一個程式如果是個人英雄主義,那麼多程式就是集體主義。(不嚴格區分多程式 和 多執行緒的差別)
你不再是一個獨行俠,而是一個指揮家。
獨來獨往,非常自由自在,但是,很多時候,不如眾人拾柴火焰高。
這就是我對多程式的理解。多執行緒程式設計的主要問題是:通訊 和 同步問題。
更多PHP 多執行緒程式設計的背景知識見:
PHP多程式程式設計(一)
在PHP 中,如果光用pcntl ,實現比較簡單的通訊問題都是很困難的。
下面介紹管道通訊:
1. 管道可以認為是一個佇列,不同的執行緒都可以往裡面寫東西,也都可以從裡面讀東西。寫就是
在佇列末尾新增,讀就是在隊頭刪除。
2. 管道一般有大小,預設一般是4K,也就是內容超過4K了,你就只能讀,不能往裡面寫了。
3. 預設情況下,管道寫入以後,就會被阻止,直到讀取他的程式讀取把資料讀完。而讀取程式也會被阻止,
直到有程式向管道寫入資料。當然,你可以改變這樣的預設屬性,用stream_set_block 函式,設定成非阻斷模式。
下面是我分裝的一個管道的類(這個類命名有問題,沒有統一,沒有時間改成統一的了,我一般先寫測試程式碼,最後分裝,所以命名上可能不統一):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 |
Code highlighting produced by Actipro CodeHighlighter (freeware)http://www.CodeHighlighter.com/--><?php class Pipe { public $fifoPath; private $w_pipe; private $r_pipe; /** * 自動建立一個管道 * * @param string $name 管道名字 * @param int $mode 管道的許可權,預設任何使用者組可以讀寫 */ function __construct($name = 'pipe', $mode = 0666) { $fifoPath = "/tmp/$name." . posix_getpid(); if (!file_exists($fifoPath)) { if (!posix_mkfifo($fifoPath, $mode)) { error("create new pipe ($name) error."); return false; } } else { error( "pipe ($name) has exit."); return false; } $this->fifoPath = $fifoPath; } /////////////////////////////////////////////////// // 寫管道函式開始 /////////////////////////////////////////////////// function open_write() { $this->w_pipe = fopen($this->fifoPath, 'w'); if ($this->w_pipe == NULL) { error("open pipe {$this->fifoPath} for write error."); return false; } return true; } function write($data) { return fwrite($this->w_pipe, $data); } function write_all($data) { $w_pipe = fopen($this->fifoPath, 'w'); fwrite($w_pipe, $data); fclose($w_pipe); } function close_write() { return fclose($this->w_pipe); } ///////////////////////////////////////////////////////// /// 讀管道相關函式開始 //////////////////////////////////////////////////////// function open_read() { $this->r_pipe = fopen($this->fifoPath, 'r'); if ($this->r_pipe == NULL) { error("open pipe {$this->fifoPath} for read error."); return false; } return true; } function read($byte = 1024) { return fread($this->r_pipe, $byte); } function read_all() { $r_pipe = fopen($this->fifoPath, 'r'); $data = ''; while (!feof($r_pipe)) { //echo "read one K\n"; $data .= fread($r_pipe, 1024); } fclose($r_pipe); return $data; } function close_read() { return fclose($this->r_pipe); } //////////////////////////////////////////////////// /** * 刪除管道 * * @return boolean is success */ function rm_pipe() { return unlink($this->fifoPath); } } ?> |
有了這個類,就可以實現簡單的管道通訊了,因為這個教程是多程式程式設計系列教程的一個部分。
這個管道類的應用部分,將放到第三部分。