PHP實現(xiàn)守護進程的示例代碼
前言
寫 PHP CLI 程序的老司機們可能經常會寫一些常駐進程,比如消息隊列消費者進程,這些進程會一直運行,除非要發(fā)版,不然一般不會重啟的,所以程序程序是不可能由我們通過 ssh 登錄到服務器上通過終端來直接啟動的(因為一旦斷開 ssh 進程就退出了),常見的做法就是用 systemd
或者 supervisor
來使其成為 守護進程,這樣進程就可以一直運行,遇到錯誤意外退出也能被自動重啟。
好學的你可能會思考守護進程到底是怎么實現(xiàn)的?為什么有的程序既可以自己就成為守護進程,又可以通過 systemd
來后臺運行?如果不依賴外部,我們的 PHP 程序該怎樣變成守護進程呢?
成為守護進程的步驟
其實只需要創(chuàng)建子進程并退出父進程,將要處理的工作在子進程中進行就可以實現(xiàn)一個守護進程了。但是僅僅是這么做的話,如果后續(xù)任務很復雜,或者引入了一些第三方包,那么可能就會出現(xiàn)奇奇怪怪的問題了。
而在《UNIX環(huán)境高級編程》(英語:Advanced Programming in the UNIX Environment,簡稱APUE)一書中有介紹關于守護進程的編碼規(guī)范,我們按照規(guī)范來實現(xiàn)我們的守護進程就可以避免出現(xiàn)那些奇怪的問題了。而且規(guī)范也不復雜,只需要幾步就可以了:
- 創(chuàng)建子進程,退出父進程
- 子進程創(chuàng)建一個新的會話并成為
session leader
- 重設文件掩碼
- 改變工作目錄
- 關閉標準輸入輸出
實現(xiàn)
<?php function daemon() { // [1] 創(chuàng)建子進程 $pid = pcntl_fork(); if ($pid == -1) { die('fork failed'); } // [2] 如果是父進程,則退出 if ($pid > 0) { exit(0); } ///////////////// 以下是子進程 ///////////////// // [3] 創(chuàng)建一個新的會話并成為 session leader if ( ($sid = posix_setsid()) <= 0 ) { die("Set sid failed.\n"); } // [4] 重設文件掩碼 umask(0); // [5] 改變工作目錄 if (chdir('/') === false) { die("chdir failed.\n"); } // [6] 關閉標準輸入輸出 fclose(STDIN); fclose(STDOUT); fclose(STDERR); } daemon(); // ... 真正的處理邏輯
說明
上面短短的十幾二十行代碼就實現(xiàn)了一個守護進程,接下來解釋一下有些步驟為什么要這么做。
創(chuàng)建子進程并退出父進程
pcntl_fork()
的返回值有三種情況,上面的代碼([1]和[2])已經處理了對應的情況。
創(chuàng)建新的會話
調用 posix_setsid()
創(chuàng)建新會話會使得當前進程成為新會話中的“會話首進程”,同時也會使當前進程成為“進程組組長”,并且使得當前進程脫離控制終端。
重設文件掩碼
調用 umask()
重設文件掩碼,這里通常是 0。為什么是 0 而不是其他呢,因為子進程從父進程繼承來的文件掩碼可能會屏蔽某些特定的文件操作權限。比如說引入的第三方庫可能需要用特定的權限來創(chuàng)建文件,并且它沒有將文件權限作為一個選項參數(shù)由你指定,那么就可能會出現(xiàn)失敗的情況;而我們傳入 0
,會使得從調用了 umask()
之后,守護進程創(chuàng)建的文件權限為 0666
,目錄權限為 0777
,均為最高權限。
關于 umask()
后面會展開新的篇幅來說明,感興趣的可以先自行搜索資料學習。
改變工作目錄
通過 chdir()
我們將工作目錄設置為根目錄 /
,主要是因為守護進程是長時間運行的,通常只有系統(tǒng)關閉/重啟才會退出。假如從父進程繼承來的工作目錄是個掛載的文件系統(tǒng),如果不改變工作目錄,那么將會導致這個掛載的文件系統(tǒng)一直沒法卸載。
當然也不一定要將工作目錄切換到根目錄,你也可以根據(jù)實際情況切換到特定的目錄。
關閉標準輸入輸出
因為守護進程是脫離終端控制的,所以是沒有標準輸入輸出交互的,我們將其關閉即可。
其他
二次 fork
你可能在一些資料中看到有人推薦你在 [3] 創(chuàng)建一個新的會話并成為 session leader 之后再次進行 fork
。這一步驟是在基于 System V
的系統(tǒng)中,可以保證你的守護進程不是“會話首進程”,可以阻止其重新申請獲取一個控制終端。
關閉不必要的文件描述符
按照編碼規(guī)范,實際還有一步是關閉不必要的文件描述符。但我們?yōu)榱撕唵纹鹨?,上面的代碼在進程啟動之后先創(chuàng)建守護進程再執(zhí)行其他操作,因此這里只打開了三個文件描述符: 0
、1
和 2
(即標準輸入、標準輸出、標準錯誤)。
注意事項
因為上面的代碼將標準輸入輸出關閉了,也就是說如果你在 daemon()
之后有諸如 echo "Hello world";
之類的輸出,那么你的程序將會出錯然后退出,并且你將看不到任何錯誤信息(因為標準錯誤也被關閉了)。
解決方案有兩種,一種是用 file_put_contents
代替 echo
,但是這樣并不優(yōu)雅,而且萬一引入的第三方包中寫了 echo
或者是 file_put_contents(STDOUT, ...)
,那你的程序也會“莫名其妙”就掛了,會讓你排查半天到底是哪里出了問題。
因此我們還可以在第[6]之后加入:
// [7] 重定向輸入輸出 global $stdin, $stdout, $stderr; $stdin = fopen('/dev/null', 'r'); $stdout = fopen('/dev/null', 'wb'); // 你也可以將標準輸出重定向到指定的文件,相當于是日志 $stderr = fopen('/dev/null', 'wb'); // 同上
到此這篇關于PHP實現(xiàn)守護進程的示例代碼的文章就介紹到這了,更多相關PHP守護進程內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
PHP syntax error, unexpected $end 錯誤的一種原因及解決
PHP 遇到 syntax error, unexpected $end 錯誤時,查錯思路其實還是看看文件里 PHP 的開始標記和結束標記是否配對,還要額外注意注釋里是否出現(xiàn)過 ?> 喲。2008-10-10PHP的array_diff()函數(shù)在處理大數(shù)組時的效率問題
PHP 5.2.6 以上版本的 array_diff() 函數(shù)在處理大數(shù)組時,需要花費超長時間,這個 bug 已經被官方確認;在這個問題被修復之前或者在我們不能控制 PHP 版本的時候,可以使用本文提供的方法2011-11-11