1. 程式人生 > >PHP多進程編程之僵屍進程問題

PHP多進程編程之僵屍進程問題

not 接收 查看 死循環 手冊 fig 再次 發現 關於

上一篇說到了使用pcntl_fork函數可以讓PHP實現多進程並發或者異步處理的效果。那麽問題是我們產生的進程需要去控制,而不能置之不理。最基本的方式就是fork進程和殺死進程。

通過利用pcntl_fork函數,我們已經有了新的子進程,而子進程接下來完成我們需要處理的內容,那麽我們就暫且叫做service()吧,而且我們需要很多個service()進行處理,再次參照我們之前的需求,父進程需要一直循環讀取配置文件,等待文件發生改變。通過對pcntl_fork的方式,很容易我們就可以寫出如下代碼:

$res = config();
//kill進程
for($i = 0; $i < $res[sum]; $i
++) { $pid = pcntl_fork(); if ($pid == 0) { service(); return; } }

代碼中註釋的地方我們需要在配置文件中發生改變的時候殺死進程,殺死進程的的方式很簡單,可以使用kill命令直接殺死,比如(假設pid為123):

1 kill 123

但是我們發現,使用這個殺死進程的方式並沒有真正的把進程殺死,這個子進程被殺死後還占用這個進程的資源,我們成為僵屍進程,僵屍進程是使用kill命令無法殺死的。想要解決這個問題,我們能做的只有兩種方式。

1. shutdown

2. 殺死該進程的父進程。

但是這兩種方法都不行,因為這個程序的目的是監控常駐在服務器內,服務器不能關閉,並且父進程也不能被幹掉。這時候我們看到了官方文檔對於fork方法的解釋:

1 pcntl_wait($status); //等待子進程中斷,防止子進程成為僵屍進程。

原來有種方式可以防止進程成為僵屍進程,但是,官網給出的代碼是這樣子的:

 1 $pid = pcntl_fork();
 2 //父進程和子進程都會執行下面代碼
 3 if ($pid == -1) {
 4     //錯誤處理:創建子進程失敗時返回-1.
 5      die(‘could not fork‘);
 6 } else if ($pid) {
7 //父進程會得到子進程號,所以這裏是父進程執行的邏輯 8 pcntl_wait($status); //等待子進程中斷,防止子進程成為僵屍進程。 9 } else { 10 //子進程得到的$pid為0, 所以這裏是子進程執行的邏輯。 11 }

什麽意思呢?就是父進程會等待子進程運行,等子進程運行結束之後,才會進行下一步,並且也會消除僵屍進程。但是這裏又和我們的需求不符合了,我們的子進程為一個死循環的程序,不斷的查找輸出,更本沒有結束的時候,並且我們需要的是異步處理而不是同步。但是這個方法可以用嗎?其實當然可以。

在pcntl_wait的文檔中是這麽解釋這個函數的:

wait函數刮起當前進程的執行直到一個子進程退出或接收到一個信號要求中斷當前進程或調用一個信號處理函數。 如果一個子進程在調用此函數時已經退出(俗稱僵屍進程),此函數立刻返回。子進程使用的所有系統資源將 被釋放。關於wait在您系統上工作的詳細規範請查看您系統的wait(2)手冊。

我們發現,當這個函數發現子進程成為了僵屍進程就會釋放僵屍進程的資源——前提是這個僵屍進程為這個父進程的子進程。那麽我們就可以巧妙的利用這個方式讓這些僵屍進程釋放資源了,所以就有了如下代碼:

1 posix_kill(123, 9);
2 pcntl_wait($status);

這樣我們先使用kill幹掉這個進程,這個進程就不會再運行了,但是這個進程成為了僵屍進程,占用著資源,我們下一句就執行一次pcntl_wait()讓這些僵屍進程釋放資源,這樣,子進程才真正的被終止了,僵屍進程被消除了。

PHP多進程編程之僵屍進程問題