当子进程挂掉时,会给父进程发送一个SIGCHLD信号,而默认SIGCHLD信号是被忽略的。
我们可以在父进程里设置一个SIGCHLD信号处理函数,在处理函数里调用wait()函数,去为子进程收尸。
void handler(int num)
{
wait();
}
但是这是有问题的。那就是有多个进程到达的情况:
假设有三个信号同时到达,第一个信号会触发信号的处理函数,其他两个信号会导致unix系统阻塞,但是并不会被缓存。
从而第二个信号被阻塞,而第三个信号被丢失了。如果其他子进程也退出了,那么他们的信号也会被丢失!
信号处理函数只调用wait一次,所以每丢失一个信号,就意味着有了一个僵尸进程。解决方法是在处理函数中调用
足够多次数的wait()去为子进程收尸!
waitpid()函数就可以解决此问题!
void handler(int num)
{
while(waitpid(-1,NULL,WNOHANG)>0);
}
waitpid提供了wait函数的超集功能。第一个参数表示他所要等待的进程的进程号。值-1表示等待所有的子进程。第二个是一个指向整型的指针,用来获取状态
不需要的话,就赋值为NULL,WNOHONG参数告诉waitpid如果没有了僵尸进程则不必等待。
函数内的循环,将会持续到所有的僵尸进程被回收为止,就算有再多的SIGCHLD信号产生,也可以被处理!!