国产成人精品亚洲777人妖,欧美日韩精品一区视频,最新亚洲国产,国产乱码精品一区二区亚洲

您的位置:首頁技術(shù)文章
文章詳情頁

php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析

瀏覽:105日期:2022-09-11 11:55:37

本文實(shí)例講述了php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法。分享給大家供大家參考,具體如下:

對于用PHP進(jìn)行多進(jìn)程并發(fā)編程,不可避免要遇到僵尸進(jìn)程的問題。

僵尸進(jìn)程是指的父進(jìn)程已經(jīng)退出,而該進(jìn)程dead之后沒有進(jìn)程接受,就成為僵尸進(jìn)程(zombie)進(jìn)程。任何進(jìn)程在退出前(使用exit退出) 都會變成僵尸進(jìn)程(用于保存進(jìn)程的狀態(tài)等信息),然后由init進(jìn)程接管。如果不及時回收僵尸進(jìn)程,那么它在系統(tǒng)中就會占用一個進(jìn)程表項(xiàng),如果這種僵尸進(jìn)程過多,最后系統(tǒng)就沒有可以用的進(jìn)程表項(xiàng),于是也無法再運(yùn)行其它的程序。

方法一:

父進(jìn)程通過pcntl_wait和pcntl_waitpid等函數(shù)等待子進(jìn)程結(jié)束

$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { //父進(jìn)程阻塞著等待子進(jìn)程的退出 //pcntl_wait($status); //pcntl_waitpid($pid, $status); //非阻塞方式 //pcntl_wait($status, WNOHANG); //pcntl_waitpid($pid, $status, WNOHANG);} else { sleep(3); echo 'child rn'; exit;}

方法二:

可以用signal函數(shù)為SIGCHLD安裝handler,因?yàn)樽舆M(jìn)程結(jié)束后,父進(jìn)程會收到該信號,可以在handler中調(diào)用pcntl_wait或pcntl_waitpid來回收。

<?phpdeclare(ticks = 1);//信號處理函數(shù)function sig_func() { echo 'SIGCHLD rn'; pcntl_wait($status); //pcntl_waitpid(-1, $status); //非阻塞 //pcntl_wait($status, WNOHANG); //pcntl_waitpid(-1, $status, WNOHANG);}pcntl_signal(SIGCHLD, ’sig_func’);$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { sleep(10);} else { sleep(3); echo 'child rn'; exit;}

如果子進(jìn)程還沒有結(jié)束時,父進(jìn)程就結(jié)束了,那么init進(jìn)程會自動接手這個子進(jìn)程,進(jìn)行回收。

如果父進(jìn)程是循環(huán),又沒有安裝SIGCHLD信號處理函數(shù)調(diào)用wait或waitpid()等待子進(jìn)程結(jié)束。那么子進(jìn)程結(jié)束后,沒有回收,就產(chǎn)生僵尸進(jìn)程了。

例如:

<?php$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { for(;;) { sleep(3); }} else { echo 'child rn'; exit;}

父進(jìn)程是個死循環(huán),也沒有安裝SIGCHLD信號處理函數(shù),子進(jìn)程結(jié)束后。我們通過如下命令查看

> ps -A -o stat,ppid,pid,cmd | grep -e ’^[Zz]’

會發(fā)現(xiàn)一個僵尸進(jìn)程。

代碼改進(jìn)一下:

<?phpdeclare(ticks = 1);//信號處理函數(shù)function sig_func() { echo 'SIGCHLD rn'; pcntl_waitpid(-1, $status, WNOHANG);}pcntl_signal(SIGCHLD, ’sig_func’);$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { for(;;) { sleep(3); }} else { echo 'child rn'; exit;}

當(dāng)子進(jìn)程結(jié)束后,再通過命令查看時,我們發(fā)現(xiàn)這時就沒有僵尸進(jìn)程了,這說明父進(jìn)程對它進(jìn)行了回收。

方法三:

如果父進(jìn)程不關(guān)心子進(jìn)程什么時候結(jié)束,那么可以用pcntl_signal(SIGCHLD, SIG_IGN)通知內(nèi)核,自己對子進(jìn)程的結(jié)束不感興趣,那么子進(jìn)程結(jié)束后,內(nèi)核會回收,并不再給父進(jìn)程發(fā)送信號。

<?phpdeclare(ticks = 1);pcntl_signal(SIGCHLD, SIG_IGN);$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { for(;;) { sleep(3); }} else { echo 'child rn'; exit;}

當(dāng)子進(jìn)程結(jié)束后,SIGCHLD信號并不會發(fā)送給父進(jìn)程,而是通知內(nèi)核對子進(jìn)程進(jìn)行了回收。

方法四:

通過pcntl_fork兩次,也就是父進(jìn)程fork出子進(jìn)程,然后子進(jìn)程中再fork出孫進(jìn)程,這時子進(jìn)程退出。那么init進(jìn)程會接管孫進(jìn)程,孫進(jìn)程退出后,init會回收。不過子進(jìn)程還是需要父進(jìn)程進(jìn)行回收。我們把業(yè)務(wù)邏輯放到孫進(jìn)程中執(zhí)行,父進(jìn)程就不需要pcntl_wait或pcntl_waitpid來等待孫進(jìn)程(即業(yè)務(wù)進(jìn)程)。

<?php$pid = pcntl_fork();if($pid == -1) { die(’fork error’);} else if ($pid) { //父進(jìn)程等待子進(jìn)程退出 pcntl_wait($status); echo 'parent rn';} else { //子進(jìn)程再fork一次,產(chǎn)生孫進(jìn)程 $cpid = pcntl_fork(); if($cpid == -1) { die(’fork error’); } else if ($cpid) { //這里是子進(jìn)程,直接退出 echo 'child rn'; exit; } else { //這里是孫進(jìn)程,處理業(yè)務(wù)邏輯 for($i = 0; $i < 10; ++$i) { echo 'work... rn'; sleep(3); } }}

子進(jìn)程退出后,父進(jìn)程回收子進(jìn)程,孫進(jìn)程繼續(xù)業(yè)務(wù)邏輯的處理。當(dāng)孫進(jìn)程也執(zhí)行完畢退出后,init回收孫進(jìn)程。

更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《PHP進(jìn)程與線程操作技巧總結(jié)》、《PHP網(wǎng)絡(luò)編程技巧總結(jié)》、《PHP基本語法入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》

希望本文所述對大家PHP程序設(shè)計(jì)有所幫助。

標(biāo)簽: PHP
相關(guān)文章:
主站蜘蛛池模板: 西宁市| 沂源县| 汝城县| 富民县| 寻乌县| 上犹县| 奉节县| 上饶市| 和平县| 定日县| 马边| 高阳县| 积石山| 株洲市| 孟州市| 高阳县| 颍上县| 三穗县| 五大连池市| 明水县| 吉安县| 马公市| 张掖市| 澜沧| 巴塘县| 正定县| 东乡族自治县| 双鸭山市| 巴彦淖尔市| 中牟县| 舒城县| 石河子市| 博湖县| 临邑县| 德州市| 从江县| 无为县| 康乐县| 西和县| 嘉峪关市| 连平县|