寫給 PHP 程序員的信號處理教程

“ 今天晚上吃滷煮,領桌的妹子問我,這玩意兒能吃麼?我:你覺得能吃就能吃。。。和內容無關的主題 

01

什麼是信號

信號就是事件發生時,對進程的一種通知機制(也叫軟件中斷)。當一個進程收到信號後,內核會暫停該進程正在執行的代碼,並跳轉到對應的信號處理函數中,如果處理函數不中斷,執行完處理函數後,會繼續執行之前中斷的地方往下執行。

我們在FPM模式下寫代碼,不會遇到信號處理相關的問題,但是CLI模式下一些常駐內存的腳本,如何能夠自由的重啓、關閉、退出前做一些清理工作(斷開鏈接,刪除臨時文件等)?

02

C的信號處理舉例

上圖中,我爲信號SIGINT註冊了處理函數sigint_handle,捕獲到信號後,輸出內容後退出,簡單易懂吧。執行  gcc -o run run.c && ./run ,然後 CTRL+C(會觸發SIGINT信號) ,成功輸出: 成功捕獲到信號2! ,程序直接結束運行。

03

PHP的信號處理舉例

pcntl_signal是PHP的信號處理註冊方法,上面實現的功能和C實現的基本一致,不同的是,當前進程不會退出,並且多輸出了一個signinfo(PHP是C寫的,爲啥剛剛C語言的沒有信號相關的信息呢?因爲PHP使用的是另一個信號函數 sigaction, 有興趣的可以瞭解一下 )

04

PHP的信號處理並不是直接調用C

這個是pcntl初始化的時候,將pcntl_signal_dispatch註冊爲tick的處理函數

pcntl_signal會將處理函數放到信號集合中(PHP的hash table),而php_signale4最終會調用sigaction進行底層的信號管理。

這裏我省略了大量代碼,將關鍵的點標記了出來,其實PHP維護一個自己的信號集合,每當調用 pcntl_signal_dispatch時就會查詢是否有信號,上面的SIG_BLOCK會將信號阻塞,這樣只有我們把關鍵的代碼執行完畢之後,再去觸發信號處理函數以保證數據和程序邏輯的完整性。

05

PHP如何優雅的處理信號

經常見到身邊的程序員們,每當需要重啓PHP-FPM進程的時候,使用的招數是kill掉所有PHP進程,然後新啓動。一般情況沒啥問題,但有些時候可能某個進程的任務還沒執行完,直接把人家中斷了略顯粗暴。其實只要你給PHP的Master進程發送一條USR2信號,它便會再處理完所有任務後,重啓子進程,這纔是所謂的優雅~

上圖是我簡單寫的一個例子,如果我們想讓進程優雅退出的時候,只需要發送SIGTERM信號即可。需要注意的是SIGKILL和SIGSTOP信號會略過信號阻塞會將進程直接停止,還有就是信號會中斷睡眠(SLEEP),sleep如果沒執行完會返回剩下的秒數,有興趣可以試試。

信號相關的知識點其實有很多,還需要繼續深入研究~上文中的PHP源碼爲7.1.25版本,各個版本可能不太一樣,如果覺得學到了點啥,順手點個好看

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章