寫給 PHP 程序員的信號處理教程
“ 今天晚上吃鹵煮,領(lǐng)桌的妹子問我,這玩意兒能吃么?我:你覺得能吃就能吃。。。和內(nèi)容無關(guān)的主題 ”
01
什么是信號
信號就是事件發(fā)生時,對進程的一種通知機制(也叫軟件中斷)。當一個進程收到信號后,內(nèi)核會暫停該進程正在執(zhí)行的代碼,并跳轉(zhuǎn)到對應(yīng)的信號處理函數(shù)中,如果處理函數(shù)不中斷,執(zhí)行完處理函數(shù)后,會繼續(xù)執(zhí)行之前中斷的地方往下執(zhí)行。
我們在FPM模式下寫代碼,不會遇到信號處理相關(guān)的問題,但是CLI模式下一些常駐內(nèi)存的腳本,如何能夠自由的重啟、關(guān)閉、退出前做一些清理工作(斷開鏈接,刪除臨時文件等)?
02
C的信號處理舉例
上圖中,我為信號SIGINT注冊了處理函數(shù)sigint_handle,捕獲到信號后,輸出內(nèi)容后退出,簡單易懂吧。執(zhí)行 gcc -o run run.c && ./run ,然后 CTRL+C(會觸發(fā)SIGINT信號) ,成功輸出: 成功捕獲到信號2! ,程序直接結(jié)束運行。
03
PHP的信號處理舉例
pcntl_signal是PHP的信號處理注冊方法,上面實現(xiàn)的功能和C實現(xiàn)的基本一致,不同的是,當前進程不會退出,并且多輸出了一個signinfo(PHP是C寫的,為啥剛剛C語言的沒有信號相關(guān)的信息呢?因為PHP使用的是另一個信號函數(shù) sigaction, 有興趣的可以了解一下 )
04
PHP的信號處理并不是直接調(diào)用C
這個是pcntl初始化的時候,將pcntl_signal_dispatch注冊為tick的處理函數(shù)
pcntl_signal會將處理函數(shù)放到信號集合中(PHP的hash table),而php_signale4最終會調(diào)用sigaction進行底層的信號管理。
這里我省略了大量代碼,將關(guān)鍵的點標記了出來,其實PHP維護一個自己的信號集合,每當調(diào)用 pcntl_signal_dispatch時就會查詢是否有信號,上面的SIG_BLOCK會將信號阻塞,這樣只有我們把關(guān)鍵的代碼執(zhí)行完畢之后,再去觸發(fā)信號處理函數(shù)以保證數(shù)據(jù)和程序邏輯的完整性。
05
PHP如何優(yōu)雅的處理信號
經(jīng)常見到身邊的程序員們,每當需要重啟PHP-FPM進程的時候,使用的招數(shù)是kill掉所有PHP進程,然后新啟動。一般情況沒啥問題,但有些時候可能某個進程的任務(wù)還沒執(zhí)行完,直接把人家中斷了略顯粗暴。其實只要你給PHP的Master進程發(fā)送一條USR2信號,它便會再處理完所有任務(wù)后,重啟子進程,這才是所謂的優(yōu)雅~
上圖是我簡單寫的一個例子,如果我們想讓進程優(yōu)雅退出的時候,只需要發(fā)送SIGTERM信號即可。需要注意的是SIGKILL和SIGSTOP信號會略過信號阻塞會將進程直接停止,還有就是信號會中斷睡眠(SLEEP),sleep如果沒執(zhí)行完會返回剩下的秒數(shù),有興趣可以試試。
信號相關(guān)的知識點其實有很多,還需要繼續(xù)深入研究~上文中的PHP源碼為7.1.25版本,各個版本可能不太一樣,如果覺得學到了點啥,順手點個好看