亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 語言 > PHP > 正文

什么是PHP7中的孤兒進程與僵尸進程

2024-05-05 00:08:24
字體:
來源:轉載
供稿:網友

基本概念

我們知道在unix/linux中,正常情況下,子進程是通過父進程創建的,子進程在創建新的進程。子進程的結束和父進程的運行是一個異步過程,即父進程永遠無法預測子進程 到底什么時候結束。 當一個 進程完成它的工作終止之后,它的父進程需要調用wait()或者waitpid()系統調用取得子進程的終止狀態。

孤兒進程

一個父進程退出,而它的一個或多個子進程還在運行,那么那些子進程將成為孤兒進程。孤兒進程將被init進程(進程號為1)所收養,并由init進程對它們完成狀態收集工作。

僵尸進程

一個進程使用fork創建子進程,如果子進程退出,而父進程并沒有調用wait或waitpid獲取子進程的狀態信息,那么子進程的進程描述符仍然保存在系統中。這種進程稱之為僵死進程。

問題及危害

unix提供了一種機制可以保證只要父進程想知道子進程結束時的狀態信息, 就可以得到。這種機制就是: 在每個進程退出的時候,內核釋放該進程所有的資源,包括打開的文件,占用的內存等。 但是仍然為其保留一定的信息(包括進程號the process ID,退出狀態the termination status of the process,運行時間the amount of CPU time taken by the process等)。直到父進程通過wait / waitpid來取時才釋放。 但這樣就導致了問題,如果進程不調用wait / waitpid的話, 那么保留的那段信息就不會釋放,其進程號就會一直被占用,但是系統所能使用的進程號是有限的,如果大量的產生僵死進程,將因為沒有可用的進程號而導致系統不能產生新的進程. 此即為僵尸進程的危害,應當避免。

孤兒進程是沒有父進程的進程,孤兒進程這個重任就落到了init進程身上,init進程就好像是一個民政局,專門負責處理孤兒進程的善后工作。每當出現一個孤兒進程的時候,內核就把孤 兒進程的父進程設置為init,而init進程會循環地wait()它的已經退出的子進程。這樣,當一個孤兒進程凄涼地結束了其生命周期的時候,init進程就會代表黨和政府出面處理它的一切善后工作。因此孤兒進程并不會有什么危害。

任何一個子進程(init除外)在exit()之后,并非馬上就消失掉,而是留下一個稱為僵尸進程(Zombie)的數據結構,等待父進程處理。這是每個 子進程在結束時都要經過的階段。如果子進程在exit()之后,父進程沒有來得及處理,這時用ps命令就能看到子進程的狀態是“Z”。如果父進程能及時 處理,可能用ps命令就來不及看到子進程的僵尸狀態,但這并不等于子進程不經過僵尸狀態。 如果父進程在子進程結束之前退出,則子進程將由init接管。init將會以父進程的身份對僵尸狀態的子進程進行處理。

僵尸進程危害場景

例如有個進程,它定期的產 生一個子進程,這個子進程需要做的事情很少,做完它該做的事情之后就退出了,因此這個子進程的生命周期很短,但是,父進程只管生成新的子進程,至于子進程 退出之后的事情,則一概不聞不問,這樣,系統運行上一段時間之后,系統中就會存在很多的僵死進程,倘若用ps命令查看的話,就會看到很多狀態為Z的進程。 嚴格地來說,僵死進程并不是問題的根源,罪魁禍首是產生出大量僵死進程的那個父進程。因此,當我們尋求如何消滅系統中大量的僵死進程時,答案就是把產生大 量僵死進程的那個元兇槍斃掉(也就是通過kill發送SIGTERM或者SIGKILL信號啦)。槍斃了元兇進程之后,它產生的僵死進程就變成了孤兒進 程,這些孤兒進程會被init進程接管,init進程會wait()這些孤兒進程,釋放它們占用的系統進程表中的資源,這樣,這些已經僵死的孤兒進程 就能瞑目而去了。

孤兒進程和僵尸進程測試

1、孤兒進程被init進程收養

$pid = pcntl_fork();if ($pid > 0) {  // 顯示父進程的進程ID,這個函數可以是getmypid(),也可以用posix_getpid()  echo "Father PID:" . getmypid() . PHP_EOL;  // 讓父進程停止兩秒鐘,在這兩秒內,子進程的父進程ID還是這個父進程  sleep(2);} else if (0 == $pid) {  // 讓子進程循環10次,每次睡眠1s,然后每秒鐘獲取一次子進程的父進程進程ID  for ($i = 1; $i <= 10; $i++) {    sleep(1);    // posix_getppid()函數的作用就是獲取當前進程的父進程進程ID    echo posix_getppid() . PHP_EOL;  }} else {  echo "fork error." . PHP_EOL;}

測試結果:

php daemo001.phpFather PID:180461804618046www@iZ2zec3dge6rwz2uw4tveuZ:~/test$ 11111111

2、僵尸進程和危害 

執行以下代碼 php zombie1.php

$pid = pcntl_fork();if( $pid > 0 ){  // 下面這個函數可以更改php進程的名稱  cli_set_process_title('php father process');  // 讓主進程休息60秒鐘  sleep(60);} else if( 0 == $pid ) {  cli_set_process_title('php child process');  // 讓子進程休息10秒鐘,但是進程結束后,父進程不對子進程做任何處理工作,這樣這個子進程就會變成僵尸進程  sleep(10);} else {  exit('fork error.'.PHP_EOL);}

執行結果,另外一個終端窗口

www@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18458 0.5 1.2 204068 25920 pts/1  S+  16:34  0:00 php father processwww   18459 0.0 0.3 204068 6656 pts/1  S+  16:34  0:00 php child processwww@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18458 0.0 1.2 204068 25920 pts/1  S+  16:34  0:00 php father processwww   18459 0.0 0.0   0   0 pts/1  Z+  16:34  0:00 [php] <defunct>

通過執行 ps -aux 命令可以看到,當程序在前十秒內運行的時候,php child process 的狀態列為 [S+],然而在十秒鐘過后,這個狀態變成了 [Z+],也就是變成了危害系統的僵尸進程。

那么,問題來了?如何避免僵尸進程呢?

PHP通過 pcntl_wait() 和 pcntl_waitpid() 兩個函數來幫我們解決這個問題。了解Linux系統編程的應該知道,看名字就知道這其實就是PHP把C語言中的 wait() 和 waitpid() 包裝了一下。

通過代碼演示 pcntl_wait() 來避免僵尸進程。

pcntl_wait() 函數:

這個函數的作用就是 “ 等待或者返回子進程的狀態 ”,當父進程執行了該函數后,就會阻塞掛起等待子進程的狀態一直等到子進程已經由于某種原因退出或者終止。

換句話說就是如果子進程還沒結束,那么父進程就會一直等等等,如果子進程已經結束,那么父進程就會立刻得到子進程狀態。這個函數返回退出的子進程的進程 ID 或者失敗返回 -1。

執行以下代碼 zombie2.php

$pid = pcntl_fork();if ($pid > 0) {  // 下面這個函數可以更改php進程的名稱  cli_set_process_title('php father process');  // 返回$wait_result,就是子進程的進程號,如果子進程已經是僵尸進程則為0  // 子進程狀態則保存在了$status參數中,可以通過pcntl_wexitstatus()等一系列函數來查看$status的狀態信息是什么  $wait_result = pcntl_wait($status);  print_r($wait_result);  print_r($status);  // 讓主進程休息60秒鐘  sleep(60);} else if (0 == $pid) {  cli_set_process_title('php child process');  // 讓子進程休息10秒鐘,但是進程結束后,父進程不對子進程做任何處理工作,這樣這個子進程就會變成僵尸進程  sleep(10);} else {  exit('fork error.' . PHP_EOL);}

在另外一個終端中通過ps -aux查看,可以看到在前十秒內,php child process 是 [S+] 狀態,然后十秒鐘過后進程消失了,也就是被父進程回收了,沒有變成僵尸進程。

www@iZ2zec3dge6rwz2uw4tveuZ:~/test$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww@iZ2zec3dge6rwz2uw4tveuZ:~/test$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18519 0.5 1.2 204068 25576 pts/1  S+  16:42  0:00 php father processwww   18520 0.0 0.3 204068 6652 pts/1  S+  16:42  0:00 php child processwww@iZ2zec3dge6rwz2uw4tveuZ:~/test$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18519 0.0 1.2 204068 25576 pts/1  S+  16:42  0:00 php father process

但是,pcntl_wait() 有個很大的問題,就是阻塞。父進程只能掛起等待子進程結束或終止,在此期間父進程什么都不能做,這并不符合多快好省原則,所以 pcntl_waitpid() 閃亮登場。pcntl_waitpid( pid, &status, $option = 0 )的第三個參數如果設置為WNOHANG,那么父進程不會阻塞一直等待到有子進程退出或終止,否則將會和pcntl_wait()的表現類似。

修改第三個案例的代碼,但是,我們并不添加WNOHANG,演示說明pcntl_waitpid()功能:

$pid = pcntl_fork();if ($pid > 0) {  // 下面這個函數可以更改php進程的名稱  cli_set_process_title('php father process');  // 返回值保存在$wait_result中  // $pid參數表示 子進程的進程ID  // 子進程狀態則保存在了參數$status中  // 將第三個option參數設置為常量WNOHANG,則可以避免主進程阻塞掛起,此處父進程將立即返回繼續往下執行剩下的代碼  $wait_result = pcntl_waitpid($pid, $status);  var_dump($wait_result);  var_dump($status);  // 讓主進程休息60秒鐘  sleep(60);} else if (0 == $pid) {  cli_set_process_title('php child process');  // 讓子進程休息10秒鐘,但是進程結束后,父進程不對子進程做任何處理工作,這樣這個子進程就會變成僵尸進程  sleep(10);} else {  exit('fork error.' . PHP_EOL);}

下面是運行結果,一個執行php zombie3.php 程序的終端窗口

www@iZ2zec3dge6rwz2uw4tveuZ:~/test$ php zombie3.phpint(18586)int(0)^C  

ctrl-c 發送 SIGINT 信號給前臺進程組中的所有進程。常用于終止正在運行的程序。

下面是ps -aux終端窗口

www@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18605 0.3 1.2 204068 25756 pts/1  S+  16:52  0:00 php father processwww   18606 0.0 0.3 204068 6636 pts/1  S+  16:52  0:00 php child processwww@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18605 0.1 1.2 204068 25756 pts/1  S+  16:52  0:00 php father processwww@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18605 0.0 1.2 204068 25756 pts/1  S+  16:52  0:00 php father processwww@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep php // ctrl-c 后不再被阻塞www@iZ2zec3dge6rwz2uw4tveuZ:~$

實際上可以看到主進程是被阻塞的,一直到第十秒子進程退出了,父進程不再阻塞  

修改第四段代碼,添加第三個參數WNOHANG,代碼如下:

$pid = pcntl_fork();if ($pid > 0) {  // 下面這個函數可以更改php進程的名稱  cli_set_process_title('php father process');  // 返回值保存在$wait_result中  // $pid參數表示 子進程的進程ID  // 子進程狀態則保存在了參數$status中  // 將第三個option參數設置為常量WNOHANG,則可以避免主進程阻塞掛起,此處父進程將立即返回繼續往下執行剩下的代碼  $wait_result = pcntl_waitpid($pid, $status, WNOHANG);  var_dump($wait_result);  var_dump($status);  echo "不阻塞,運行到這里" . PHP_EOL;  // 讓主進程休息60秒鐘  sleep(60);} else if (0 == $pid) {  cli_set_process_title('php child process');  // 讓子進程休息10秒鐘,但是進程結束后,父進程不對子進程做任何處理工作,這樣這個子進程就會變成僵尸進程  sleep(10);} else {  exit('fork error.' . PHP_EOL);}

執行 php zombie4.php

www@iZ2zec3dge6rwz2uw4tveuZ:~/test$ php zombie4.phpint(0)int(0)不阻塞,運行到這里 

另一個ps -aux終端窗口

www@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18672 0.3 1.2 204068 26284 pts/1  S+  17:00  0:00 php father processwww   18673 0.0 0.3 204068 6656 pts/1  S+  17:00  0:00 php child processwww@iZ2zec3dge6rwz2uw4tveuZ:~$ ps -aux|grep -v "grep/|nginx/|php-fpm" | grep phpwww   18672 0.0 1.2 204068 26284 pts/1  S+  17:00  0:00 php father processwww   18673 0.0 0.0   0   0 pts/1  Z+  17:00  0:00 [php] <defunct>

實際上可以看到主進程是被阻塞的,一直到第十秒子進程退出了,父進程不再阻塞?! ?/p>

問題出現了,竟然php child process進程狀態竟然變成了[Z+],這是怎么搞得?回頭分析一下代碼:
我們看到子進程是睡眠了十秒鐘,而父進程在執行pcntl_waitpid()之前沒有任何睡眠且本身不再阻塞,所以,主進程自己先執行下去了,而子進程在足足十秒鐘后才結束,進程狀態自然無法得到回收。

如果我們將代碼修改一下,就是在主進程的pcntl_waitpid()前睡眠15秒鐘,這樣就可以回收子進程了。但是即便這樣修改,細心想的話還是會有個問題,那就是在子進程結束后,在父進程執行pcntl_waitpid()回收前,有五秒鐘的時間差,在這個時間差內,php child process也將會是僵尸進程。那么,pcntl_waitpid()如何正確使用啊?這樣用,看起來畢竟不太科學。

那么,是時候引入信號學了!


注:相關教程知識閱讀請移步到PHP教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
中文字幕亚洲欧美一区二区三区| 欧美亚洲国产成人精品| 午夜精品久久久久久99热软件| 欧美性猛交丰臀xxxxx网站| 国产一区二区三区视频| 国产91ⅴ在线精品免费观看| 精品国产欧美一区二区五十路| 性色av一区二区咪爱| 国产精品久久久久久久久免费看| 欧美日韩午夜激情| 国产97在线|亚洲| 日av在线播放中文不卡| 国产美女精品视频免费观看| 97超视频免费观看| 日韩高清av一区二区三区| 日韩av在线免费播放| 久久精品视频导航| 91国产中文字幕| 自拍偷拍亚洲一区| 国产精品久久久久久久久久小说| 成人精品一区二区三区电影黑人| 成人a视频在线观看| 精品一区二区三区电影| 国产91精品视频在线观看| 欧美性色xo影院| 亚洲欧洲日产国产网站| 91av中文字幕| 久青草国产97香蕉在线视频| 亚洲最大福利视频网| 国产精品xxx视频| 在线中文字幕日韩| 久99九色视频在线观看| 91精品中国老女人| 国产欧美中文字幕| 九九综合九九综合| 麻豆乱码国产一区二区三区| 亚洲美女性视频| 欧美视频中文在线看| 久久91亚洲人成电影网站| 国产精品福利无圣光在线一区| 亚洲天堂日韩电影| 久久精品国产69国产精品亚洲| 亚洲视频777| 亚洲精品视频二区| 亚洲精品日韩欧美| 一区二区三区美女xx视频| 成人免费看吃奶视频网站| 精品亚洲一区二区三区在线观看| 国产97在线|日韩| 精品一区二区三区四区在线| 亚洲国产毛片完整版| 欧美精品激情在线观看| 国产欧美最新羞羞视频在线观看| 福利视频第一区| 久久久久久九九九| 日韩av在线精品| 国产精品中文久久久久久久| 国产精品人成电影在线观看| 日韩欧美高清视频| 国产精品精品久久久| 欧美成人全部免费| 国产精品视频一| 日本欧美一二三区| 日韩中文字幕视频| 日本高清视频精品| 亚洲www永久成人夜色| 精品精品国产国产自在线| 国产精彩精品视频| 日韩黄在线观看| 欧美国产欧美亚洲国产日韩mv天天看完整| 中文字幕在线看视频国产欧美在线看完整| 亚洲精品日韩欧美| 欧日韩在线观看| 欧美专区第一页| 欧美黑人极品猛少妇色xxxxx| 日韩经典第一页| 国产在线观看不卡| 欧美日本在线视频中文字字幕| 亚洲精品日韩丝袜精品| 一区二区三区日韩在线| 国产亚洲人成网站在线观看| 亚洲精品视频播放| 日韩专区中文字幕| 亚洲白拍色综合图区| 不卡av电影院| 日韩精品高清在线观看| 成人久久久久久| 国产一区香蕉久久| 亚洲国产精品久久久久秋霞不卡| 中文字幕日韩精品有码视频| 欧美一级淫片videoshd| 国产又爽又黄的激情精品视频| 懂色av中文一区二区三区天美| 日韩精品福利网站| 亚洲精品在线视频| 日韩小视频网址| 欧美一区二三区| 91色视频在线导航| 91精品国产综合久久香蕉的用户体验| 国产91在线高潮白浆在线观看| 国产精品视频午夜| 久久久91精品国产一区不卡| 亚洲国产精品久久久久久| 亚洲精品wwww| 久久视频免费观看| 亚洲成人网在线观看| 国产一区二区三区在线免费观看| 欧美精品在线观看| 亚洲综合日韩在线| 日韩国产欧美精品一区二区三区| 96国产粉嫩美女| 亚洲精品理论电影| 日韩网站免费观看高清| 91久久精品国产91久久性色| 成人激情视频在线观看| 日韩在线不卡视频| 91在线免费视频| 精品偷拍各种wc美女嘘嘘| 在线观看国产精品91| 91精品视频观看| 中文字幕亚洲欧美日韩在线不卡| 欧美日韩午夜剧场| 91色在线视频| 亚洲天堂男人天堂| 成人xxxx视频| 久久久久这里只有精品| 国产成人鲁鲁免费视频a| 久久久久这里只有精品| 亚洲免费视频一区二区| 欧美在线视频免费| 欧美成人三级视频网站| 欧美日韩精品在线视频| 国产精品久久不能| 久久精品国产精品亚洲| 久久久久亚洲精品成人网小说| 精品视频久久久| 亚洲天堂色网站| 茄子视频成人在线| 亚洲护士老师的毛茸茸最新章节| 国产欧美精品一区二区| 日韩精品欧美国产精品忘忧草| 久久久精品网站| 欧美激情亚洲自拍| 欧美精品免费看| 日韩精品亚洲视频| 日韩免费看的电影电视剧大全| 成人美女免费网站视频| 高跟丝袜欧美一区| 国产精品视频久久久| 日韩在线中文字| 亚洲精品电影在线| 久久99精品视频一区97| 91青草视频久久| 亚洲精品视频免费| 亚洲精品国产精品国自产在线| 国产精品久久电影观看| 久久精品国产欧美激情| 日本精品性网站在线观看| 欧美国产第二页| 国产精品自拍视频| 国产精品视频男人的天堂| 色偷偷888欧美精品久久久| 欧美日韩国产页| 欧洲日本亚洲国产区|