僵尸进程是当子进程比父进程先结束,而父进程又没有回收子进程,释放子进程占用的资源,此时子进程将成为一个僵尸进程。
在unix进程管理中,如果你新开的子进程运行结束,父进程将会收到一个SIGCHLD信号,子进程成为僵尸进程(保存了进程的状态等信息),等待父进程的处理,如果父进程一直不处理,该进程将会一直存在,占用系统进程表项,如果僵尸进程过多,导致系统没有可用的进程表项,于是再也无法运行其他的程序
为了更容易理解,本文使用pcntl扩展进行进程管理
例如:
<?php
$num = 1;
$str = "EasySwoole,Easy学swoole
";
$pid = pcntl_fork();
if ($pid > 0) {//主进程代码
echo "我是主进程,id是".getmypid().",子进程的pid是{$pid}
";
pcntl_async_signals(true);
pcntl_signal(SIGCHLD, function () {
echo '子进程退出了,请及时处理' . PHP_EOL;
});
while (1) {//主进程一直不退出
sleep(1);
}
} elseif ($pid == 0) {
echo "我是子进程,我的pid是" . getmypid() . "
";
} else {
echo "我是主进程,我慌得一批,开启子进程失败了
";
}
使用ps查看僵尸进程:
ps -A -ostat,ppid,pid,cmd |grep -e '^[Zz]'
输出:
Z+ 7136 7137 [php] <defunct>
当主进程退出之后,子进程将会被init接管并处理
回收僵尸进程
回收僵尸进程
通过pcntl_wait和pcntl_waitpid等函数等待子进程结束
<?php
$pid = pcntl_fork();
if ($pid == -1) {
die('fork error');
} else if ($pid > 0) {
//父进程阻塞着等待子进程的退出
// pcntl_wait($status);
// pcntl_waitpid($pid, $status);
//非阻塞方式
// pcntl_wait($status, WNOHANG);
// pcntl_waitpid($pid, $status, WNOHANG);
} else {
sleep(3);
echo "child
";
exit;
}
通过signal函数为SIGCHLD安装handler,因为子进程结束后,父进程会收到该信号,可以在handler中调用pcntl_wait或pcntl_waitpid来回收.
<?php
pcntl_async_signals(true);
pcntl_signal(SIGCHLD, function () {
echo "SIGCHLD
";
//阻塞方式
pcntl_wait($status);
//pcntl_waitpid(-1, $status);
//非阻塞
//pcntl_wait($status, WNOHANG);
//pcntl_waitpid(-1, $status, WNOHANG);
});
$pid = pcntl_fork();
if ($pid == -1) {
die('fork error');
} else if ($pid) {
sleep(10);
} else {
sleep(3);
echo "child
";
exit;
}
忽略掉子进程结束信号,交给init进程管理
<?php
pcntl_async_signals(true);
pcntl_signal(SIGCHLD, SIG_IGN);
$pid = pcntl_fork();
if ($pid == -1) {
die('fork error');
} else if ($pid>0) {
while(1){
sleep(1);
}
} else {
sleep(3);
echo "child
";
exit;
}