僵尸进程是当子进程比父进程先结束,而父进程又没有回收子进程,释放子进程占用的资源,此时子进程将成为一个僵尸进程。
在unix进程管理中,如果你新开的子进程运行结束,父进程将会收到一个SIGCHLD信号,子进程成为僵尸进程(保存了进程的状态等信息),等待父进程的处理,如果父进程一直不处理,该进程将会一直存在,占用系统进程表项,如果僵尸进程过多,导致系统没有可用的进程表项,于是再也无法运行其他的程序
为了更容易理解,本文使用pcntl扩展进行进程管理
例如:
<?php
$num = 1; $str = "EasySwoole,Easy学swoole\n"; $pid = pcntl_fork(); if ($pid > 0) {//主进程代码 echo "我是主进程,id是".getmypid().",子进程的pid是{$pid}\n"; pcntl_async_signals(true); pcntl_signal(SIGCHLD, function () { echo '子进程退出了,请及时处理' . PHP_EOL; }); while (1) {//主进程一直不退出 sleep(1); } } elseif ($pid == 0) { echo "我是子进程,我的pid是" . getmypid() . "\n"; } else { echo "我是主进程,我慌得一批,开启子进程失败了\n"; }
使用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 \r\n"; exit; }
通过signal函数为SIGCHLD安装handler,因为子进程结束后,父进程会收到该信号,可以在handler中调用pcntl_wait或pcntl_waitpid来回收.
<?php
pcntl_async_signals(true); pcntl_signal(SIGCHLD, function () { echo "SIGCHLD \r\n"; //阻塞方式 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 \r\n"; 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 \r\n"; exit; }