为什么有僵尸进程
设置僵死状态的目的是维护子进程的信息,以便父进程在以后某个时候获取。这些信息至少包括进程ID,进程的终止状态,以及该进程使用的CPU时间,所以当终止子进程的父进程调用wait或waitpid时就可以得到这些信息。
有些unix系统在ps命令输出的COMMAND栏以defunct指示僵死进程。
什么是僵死进程
当进程死亡时,内核会释放终止进程(调用了exit系统调用)所使用的所有存储区,关闭所有打开的文件等,但内核为每一个终止子进程保存了一定量的信息。这些信息至少包括进程ID,进程的终止状态,以及该进程使用的CPU时间,所以当终止子进程的父进程调用wait或waitpid时就可以得到这些信息。
处理僵死进程
内核在任何一个进程终止时会发给它的父进程一个信号,这个信号就是SIGCHLD。
所以,建立俘获SIGCHLD信号的信号处理函数来处理僵死进程
void sig_chld(int signo) {
pid_t pid;
int stat;
pid = wait(&stat);
}
但是,wait有个问题,如果没有已终止的子进程,却有一个或多个子进程仍在进行,wait将阻塞到有一个子进程终止为止。
waitpid则不会。
/* 成功返回终止的子进程pid,失败返回-1或者0 */
pid_t waitpid(pid_t pid, int *statloc, int options);
所以,修正后的sig_chld函数是
void sig_child(int signo) {
pid_t pid;
int stat;
while ((pid = waitpid(-1, &stat, WNOHANG)) > 0);
}
while循环是获取所有已终止子进程的状态,值-1表示等待第一个终止的子进程,WHNOHANG表示当没有已经终止的子进程时不要阻塞。
最后,在listen监听后,调用signal(SIGCHLD, sig_child),就避免出现僵死进程。
参考
[1] https://www.cnblogs.com/wuchanming/p/4020463.html
[2] 《unix网络编程》第5.9章节