php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法分析
本文實(shí)例講述了php多進(jìn)程并發(fā)編程防止出現(xiàn)僵尸進(jìn)程的方法。分享給大家供大家參考,具體如下:
對(duì)于用PHP進(jìn)行多進(jìn)程并發(fā)編程,不可避免要遇到僵尸進(jìn)程的問(wèn)題。
僵尸進(jìn)程是指的父進(jìn)程已經(jīng)退出,而該進(jìn)程dead之后沒有進(jìn)程接受,就成為僵尸進(jìn)程(zombie)進(jìn)程。任何進(jìn)程在退出前(使用exit退出) 都會(huì)變成僵尸進(jìn)程(用于保存進(jìn)程的狀態(tài)等信息),然后由init進(jìn)程接管。如果不及時(shí)回收僵尸進(jìn)程,那么它在系統(tǒng)中就會(huì)占用一個(gè)進(jìn)程表項(xiàng),如果這種僵尸進(jìn)程過(guò)多,最后系統(tǒng)就沒有可以用的進(jìn)程表項(xiàng),于是也無(wú)法再運(yùn)行其它的程序。
方法一:
父進(jìn)程通過(guò)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 \r\n";
exit;
}
方法二:
可以用signal函數(shù)為SIGCHLD安裝handler,因?yàn)樽舆M(jìn)程結(jié)束后,父進(jìn)程會(huì)收到該信號(hào),可以在handler中調(diào)用pcntl_wait或pcntl_waitpid來(lái)回收。
<?php
declare(ticks = 1);
//信號(hào)處理函數(shù)
function sig_func() {
echo "SIGCHLD \r\n";
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 \r\n";
exit;
}
如果子進(jìn)程還沒有結(jié)束時(shí),父進(jìn)程就結(jié)束了,那么init進(jìn)程會(huì)自動(dòng)接手這個(gè)子進(jìn)程,進(jìn)行回收。
如果父進(jìn)程是循環(huán),又沒有安裝SIGCHLD信號(hào)處理函數(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 \r\n";
exit;
}
父進(jìn)程是個(gè)死循環(huán),也沒有安裝SIGCHLD信號(hào)處理函數(shù),子進(jìn)程結(jié)束后。我們通過(guò)如下命令查看
> ps -A -o stat,ppid,pid,cmd | grep -e '^[Zz]'
會(huì)發(fā)現(xiàn)一個(gè)僵尸進(jìn)程。
代碼改進(jìn)一下:
<?php
declare(ticks = 1);
//信號(hào)處理函數(shù)
function sig_func() {
echo "SIGCHLD \r\n";
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 \r\n";
exit;
}
當(dāng)子進(jìn)程結(jié)束后,再通過(guò)命令查看時(shí),我們發(fā)現(xiàn)這時(shí)就沒有僵尸進(jìn)程了,這說(shuō)明父進(jìn)程對(duì)它進(jìn)行了回收。
方法三:
如果父進(jìn)程不關(guān)心子進(jìn)程什么時(shí)候結(jié)束,那么可以用pcntl_signal(SIGCHLD, SIG_IGN)通知內(nèi)核,自己對(duì)子進(jìn)程的結(jié)束不感興趣,那么子進(jìn)程結(jié)束后,內(nèi)核會(huì)回收,并不再給父進(jìn)程發(fā)送信號(hào)。
<?php
declare(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 \r\n";
exit;
}
當(dāng)子進(jìn)程結(jié)束后,SIGCHLD信號(hào)并不會(huì)發(fā)送給父進(jìn)程,而是通知內(nèi)核對(duì)子進(jìn)程進(jìn)行了回收。
方法四:
通過(guò)pcntl_fork兩次,也就是父進(jìn)程fork出子進(jìn)程,然后子進(jìn)程中再fork出孫進(jìn)程,這時(shí)子進(jìn)程退出。那么init進(jìn)程會(huì)接管孫進(jìn)程,孫進(jìn)程退出后,init會(huì)回收。不過(guò)子進(jìn)程還是需要父進(jìn)程進(jìn)行回收。我們把業(yè)務(wù)邏輯放到孫進(jìn)程中執(zhí)行,父進(jìn)程就不需要pcntl_wait或pcntl_waitpid來(lái)等待孫進(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 \r\n";
} else {
//子進(jìn)程再fork一次,產(chǎn)生孫進(jìn)程
$cpid = pcntl_fork();
if($cpid == -1) {
die('fork error');
} else if ($cpid) {
//這里是子進(jìn)程,直接退出
echo "child \r\n";
exit;
} else {
//這里是孫進(jìn)程,處理業(yè)務(wù)邏輯
for($i = 0; $i < 10; ++$i) {
echo "work... \r\n";
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基本語(yǔ)法入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫(kù)操作入門教程》及《php常見數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家PHP程序設(shè)計(jì)有所幫助。
相關(guān)文章
利用discuz自帶通行證整合dedecms的方法以及文件下載
利用discuz自帶通行證整合dedecms的方法以及文件下載...2007-03-03
php中文語(yǔ)義分析實(shí)現(xiàn)方法示例
這篇文章主要介紹了php中文語(yǔ)義分析實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了PHP基于BosonNLP擴(kuò)展實(shí)現(xiàn)中文語(yǔ)義分析的具體操作步驟與相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-09-09
php+mysql結(jié)合Ajax實(shí)現(xiàn)點(diǎn)贊功能完整實(shí)例
這篇文章主要介紹了php+mysql結(jié)合Ajax實(shí)現(xiàn)點(diǎn)贊功能,以一個(gè)完整實(shí)例形式詳細(xì)分析了實(shí)現(xiàn)點(diǎn)贊功能中涉及的html頁(yè)面、Ajax功能及php方法的使用技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-01-01
詳解WordPress開發(fā)中的get_post與get_posts函數(shù)使用
這篇文章主要介紹了WordPress開發(fā)中的get_post與get_posts函數(shù)使用,其中一般使用get_posts()函數(shù)來(lái)返回文章數(shù)組而較少使用get_post(),需要的朋友可以參考下2016-01-01
php自動(dòng)給文章加關(guān)鍵詞鏈接的函數(shù)代碼
自動(dòng)給文章加關(guān)鍵詞鏈接的php函數(shù)代碼,需要的朋友可以參考下2012-11-11
Dedecms V3.1 生成HTML速度的優(yōu)化辦法
Dedecms V3.1 生成HTML速度的優(yōu)化辦法...2007-03-03
PHP中設(shè)置時(shí)區(qū),記錄日志文件的實(shí)現(xiàn)代碼
php中設(shè)置時(shí)區(qū) 記錄日志文件的實(shí)現(xiàn)代碼,需要的朋友可以參考下2013-01-01

