PHP語言之如何優雅的殺死PHP的死循環(常駐進程)


摘要:本文主要向大家介紹了PHP語言的如何優雅的殺死PHP的死循環(常駐進程),通過具體的實例向大家展示,希望對大家學習php語言有所幫助。

本文主要向大家介紹了PHP語言的如何優雅的殺死PHP的死循環(常駐進程),通過具體的實例向大家展示,希望對大家學習php語言有所幫助。

在日常開發的過程中,遇到一個問題,就是

<?phpwhile(true) {
    ....
    ....
    do something...
    sleep(6);
}

有這么一個常駐進程,每隔6秒,干一些事情,這個時候,突然出了一些意外狀況或者要迭代版本,得先給這個進程停掉,打斷循環退出。如果直接kill掉,那么do something可能做了一半,然后直接被殺死了,就會導致意外的bug。

方法一

用一個比較low的方法

首先創建一個test.txt的文件,里面內容為空

<?phpwhile(true) {
    if(file_get_contents('test.txt') == ''){
        break;
    }
    ....
    ....
    do something...
    sleep(6);
}

然后,如果要退出這個死循環,又保證都處理完了,那么只需要,直接在test.txt中寫點東西,就OK了

echo "done" > test.txt

利用信號量

按着第一個方法的思路,將文件換為信號量

declare(ticks=1);//配合信號量殺死進程用

$killed=false;// 注冊一個 SIGUSR2 的信號處理器

pcntl_signal(SIGUSR2, function ($signo) {
     global $killed;
     if ($signo == SIGUSR2) {
         $killed = true;
     }
 });
while(true){
    if($killed){
        break;
    }
    ....
    ....
    do something...
    sleep(6);
}

有沒有感覺高大上很多。

要想殺掉這個進程,只需要

kill -SIGUSR2 進程id

1

關於信號量,這里就補充一下吧

~$ kill -l

 1) SIGHUP         2) SIGINT         3) SIGQUIT        4) SIGILL

 5) SIGTRAP        6) SIGABRT        7) SIGBUS         8) SIGFPE

 9) SIGKILL       10) SIGUSR1       11) SIGSEGV       12) SIGUSR213) SIGPIPE       14) SIGALRM       15) SIGTERM       17) SIGCHLD18) SIGCONT       19) SIGSTOP       20) SIGTSTP       21) SIGTTIN22) SIGTTOU       23) SIGURG        24) SIGXCPU       25) SIGXFSZ26) SIGVTALRM     27) SIGPROF       28) SIGWINCH      29) SIGIO30) SIGPWR        31) SIGSYS        34) SIGRTMIN      35) SIGRTMIN+136) SIGRTMIN+2    37) SIGRTMIN+3    38) SIGRTMIN+4    39) SIGRTMIN+540) SIGRTMIN+6    41) SIGRTMIN+7    42) SIGRTMIN+8    43) SIGRTMIN+944) SIGRTMIN+10   45) SIGRTMIN+11   46) SIGRTMIN+12   47) SIGRTMIN+1348) SIGRTMIN+14   49) SIGRTMIN+15   50) SIGRTMAX-14   51) SIGRTMAX-1352) SIGRTMAX-12   53) SIGRTMAX-11   54) SIGRTMAX-10   55) SIGRTMAX-956) SIGRTMAX-8    57) SIGRTMAX-7    58) SIGRTMAX-6    59) SIGRTMAX-560) SIGRTMAX-4    61) SIGRTMAX-3    62) SIGRTMAX-2    63) SIGRTMAX-164) SIGRTMAX

信號的詳細說明:

1) SIGHUP 本信號在用戶終端連接(正常或非正常)結束時發出, 通常是在終端的控制進程結束時, 通知同一session內的各個作業, 這時它們與控制終端不再關聯.

2) SIGINT 程序終止(interrupt)信號, 在用戶鍵入INTR字符(通常是Ctrl-C)時發出。

3) SIGQUIT 和SIGINT類似, 但由QUIT字符(通常是Ctrl-)來控制. 進程在因收到SIGQUIT退出時會產生core文件, 在這個意義上類似於一個程序錯誤信號.

4) SIGILL 執行了非法指令. 通常是因為可執行文件本身出現錯誤, 或者試圖執行數據段. 堆棧溢出時也有可能產生這個信號.

5) SIGTRAP 由斷點指令或其它trap指令產生. 由debugger使用.

6) SIGABRT 程序自己發現錯誤並調用abort時產生.

6) SIGIOT 在PDP-11上由iot指令產生, 在其它機器上和SIGABRT一樣.

7) SIGBUS 非法地址, 包括內存地址對齊(alignment)出錯. eg: 訪問一個四個字長 的整數, 但其地址不是4的倍數.

8) SIGFPE 在發生致命的算術運算錯誤時發出. 不僅包括浮點運算錯誤, 還包括溢出及除數為0等其它所有的算術的錯誤.

9) SIGKILL 用來立即結束程序的運行. 本信號不能被阻塞, 處理和忽略.

10) SIGUSR1 留給用戶使用

11) SIGSEGV 試圖訪問未分配給自己的內存, 或試圖往沒有寫權限的內存地址寫數據.

12) SIGUSR2 留給用戶使用

13) SIGPIPE Broken pipe

14) SIGALRM 時鍾定時信號, 計算的是實際的時間或時鍾時間. alarm函數使用該信號.

15) SIGTERM 程序結束(terminate)信號, 與SIGKILL不同的是該信號可以被阻塞和處理. 通常用來要求程序自己正常退出. shell命令kill缺省產生這個信號.

17) SIGCHLD 子進程結束時, 父進程會收到這個信號.

18) SIGCONT 讓一個停止(stopped)的進程繼續執行. 本信號不能被阻塞. 可以用 一個handler來讓程序在由stopped狀態變為繼續執行時完成特定的工作. 例如, 重新顯示提示符。

19) SIGSTOP 停止(stopped)進程的執行. 注意它和terminate以及interrupt的區別: 該進程還未結束, 只是暫停執行. 本信號不能被阻塞, 處理或忽略.

20) SIGTSTP 停止進程的運行, 但該信號可以被處理和忽略. 用戶鍵入SUSP字符時 (通常是Ctrl-Z)發出這個信號。

21) SIGTTIN 當后台作業要從用戶終端讀數據時, 該作業中的所有進程會收到SIGTTIN 信號. 缺省時這些進程會停止執行.

22) SIGTTOU 類似於SIGTTIN, 但在寫終端(或修改終端模式)時收到.

23) SIGURG 有”緊急”數據或out-of-band數據到達socket時產生.

24) SIGXCPU 超過CPU時間資源限制. 這個限制可以由getrlimit/setrlimit來讀取/ 改變

25) SIGXFSZ 超過文件大小資源限制.

26) SIGVTALRM 虛擬時鍾信號. 類似於SIGALRM, 但是計算的是該進程占用的CPU時間.

27) SIGPROF 類似於SIGALRM/SIGVTALRM, 但包括該進程用的CPU時間以及系統調用的時間.

28) SIGWINCH 窗口大小改變時發出.

29) SIGIO 文件描述符准備就緒, 可以開始進行輸入/輸出操作.

30) SIGPWR Power failure

有兩個信號可以停止進程:SIGTERM和SIGKILL。 SIGTERM比較友好,進程能捕捉這個信號,根據您的需要來關閉程序。在關閉程序之前,您可以結束打開的記錄文件和完成正在做的任務。在某些情況下,假 如進程正在進行作業而且不能中斷,那么進程可以忽略這個SIGTERM信號。

對於SIGKILL信號,進程是不能忽略的。這是一個 “我不管您在做什么,立刻停止”的信號。假如您發送SIGKILL信號給進程,Linux就將進程停止在那里。

以上就介紹了PHP的相關知識,希望對PHP有興趣的朋友有所幫助。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM