abstract:準備我們都知道PHP是單進程執(zhí)行的,PHP處理多并發(fā)主要是依賴服務器或PHP-FPM的多進程及它們進程的復用,但PHP實現(xiàn)多進程也意義重大,尤其是在后臺Cli模式下處理大量數(shù)據(jù)或運行后臺DEMON守護進程時,多進程的優(yōu)勢不用多說。PHP的多線程也曾被人提及,但進程內(nèi)多線程資源共享和分配的問題難以解決。PHP也有多線程想關(guān)的擴展 pthreads ,但據(jù)說不太穩(wěn)定,且要求環(huán)境為線程安全,所用不多。以
準備
我們都知道PHP是單進程執(zhí)行的,PHP處理多并發(fā)主要是依賴服務器或PHP-FPM的多進程及它們進程的復用,但PHP實現(xiàn)多進程也意義重大,尤其是在后臺Cli模式下處理大量數(shù)據(jù)或運行后臺DEMON守護進程時,多進程的優(yōu)勢不用多說。
PHP的多線程也曾被人提及,但進程內(nèi)多線程資源共享和分配的問題難以解決。PHP也有多線程想關(guān)的擴展 pthreads ,但據(jù)說不太穩(wěn)定,且要求環(huán)境為線程安全,所用不多。
以前PHP群里的一位大神曾指導說后臺PHP想進階必然避不開多進程,正好公司里的守護進程也應用了PHP的多進程,結(jié)合著谷哥的各種資料和手冊,總算理解了多進程,并自己寫了一個小demo(在linux系統(tǒng)上實現(xiàn)的),用此文總結(jié)一下,如有錯漏,謝謝提出。
要實現(xiàn)PHP的多進程,我們需要兩個擴展 pcntl和 posix,安裝方法這里不再贅述。
創(chuàng)建子進程
創(chuàng)建PHP子進程是多進程的開始,我們需要pcntl_fork()函數(shù);
fork函數(shù)詳解
pcntl_fork() — 在當前進程當前位置產(chǎn)生分支(子進程)。此函數(shù)創(chuàng)建了一個新的子進程后,子進程會繼承父進程當前的上下文,和父進程一樣從pcntl_fork()函數(shù)處繼續(xù)向下執(zhí)行,只是獲取到的pcntl_fork()的返回值不同,我們便能從判斷返回值來區(qū)分父進程和子進程,分配父進程和子進程去做不同的邏輯處理。
pcntl_fork()函數(shù)成功執(zhí)行時會在父進程返回子進程的進程id(pid),因為系統(tǒng)的初始進程init進程的pid為1,后來產(chǎn)生進程的pid都會大于此進程,所以我們可以通過判斷pcntl_fork()的返回值大于1來確實當前進程是父進程;
而在子進程中,此函數(shù)的返回值會是固定值0,我們也可以通過判斷pcntl_fork()的返回值為0來確定子進程;
而pcntl_fork()函數(shù)在執(zhí)行失敗時,會在父進程返回-1,當然也不會有子進程產(chǎn)生。
fork進程實例
以下是fork子進程的一個簡單的小例子:
$ppid = posix_getpid(); $pid = pcntl_fork(); if ($pid == -1) { throw new Exception('fork子進程失敗!'); } elseif ($pid > 0) { cli_set_process_title("我是父進程,我的進程id是{$ppid}."); sleep(30); // 保持30秒,確保能被ps查到 } else { $cpid = posix_getpid(); cli_set_process_title("我是{$ppid}的子進程,我的進程id是{$cpid}."); sleep(30); }
這時介紹一下兩個函數(shù):
posix_getpid():獲取當前進程的pid;
cli_set_process_title('響亮的名字'):為當前進程取一個響亮的名字。
運行這個例子,我們便能看到當前兩個PHP進程了。
管理子進程
創(chuàng)建好了進程,那么怎么對子進程進行管理呢?使用信號。
在計算機科學中,信號是Unix、類Unix以及其他POSIX兼容的操作系統(tǒng)中進程間通訊的一種有限制的方式。它是一種異步的通知機制,用來提醒進程一個事件已經(jīng)發(fā)生。
分發(fā)信號處理器
我們通過在父進程接收子進程傳來的信號,判斷子進程狀態(tài),來對子進程進行管理。
我們需要在父進程里使用pcntl_signal()函數(shù)和pcntl_signal_dispatch()函數(shù)來給各個子進程安裝信號處理器。
pcntl_signal (int $signo , callback $handler) 安裝一個信號處理器; $signo是待處理的信號常量,callback是其處理函數(shù) pcntl_signal_dispatch () 調(diào)用每個等待信號通過pcntl_signal()安裝的處理器
PHP內(nèi)常見的信號常量有:
SIGCHLD 子進程退出成為僵尸進程會向父進程發(fā)送此信號 SIGHUP 進程掛起 SIGTEM 進程終止 ... // 其他請在手冊中查看
安裝并調(diào)用信號處理器后,一旦子進程有相應的信號返回給父進程,父進程就可以調(diào)用相應的callback函數(shù)對子進程處理;
處理子進程
對子進程的處理方法有:
posix_kill():此函數(shù)并不能顧名思義,它通過向子進程發(fā)送一個信號來操作子進程,在需要要時可以選擇給子進程發(fā)送進程終止信號來終止子進程;
pcntl_waitpid():等待或返回fork的子進程狀態(tài),如果指定的子進程在此函數(shù)調(diào)用時已經(jīng)退出(俗稱僵尸進程),此函數(shù)將立刻返回,并釋放子進程的所有系統(tǒng)資源,此進程可以避免子進程變成僵尸進程,造成系統(tǒng)資源浪費;
下面是兩個函數(shù)的函數(shù)原型:
bool posix_kill ( int $pid , int $sig ) // 向進程id為$pid的進程發(fā)送$sig信號,$sig常見信號如上; int pcntl_waitpid ( int $pid , int &$status [, int $options = 0 ] ) // 掛起當前進程的執(zhí)行直到進程號為$pid的進程退出(如果$pid為-1,則等待任意一個子進程);
總結(jié)
這就是PHP多進程的基礎(chǔ)使用了,感興趣的可以自己寫一個demo試一試手了。
最后貼一下鳥哥所說的PHP多進程優(yōu)點:
使用多進程, 子進程結(jié)束以后, 內(nèi)核會負責回收資源
使用多進程,子進程異常退出不會導致整個進程Thread退出. 父進程還有機會重建流程.
一個常駐主進程, 只負責任務分發(fā), 邏輯更清楚.