linux僵尸进程
什么是僵尸进程?
在UNIX 系统中,一个进程结束了,但是他的父进程没有等待(调用wait / waitpid)他, 那么他将变成一个僵尸进程. 在fork()/execve()过程中,假设子进程结束时父进程仍存在,而父进程fork()之前既没安装SIGCHLD信号处理函数调用 waitpid()等待子进程结束,又没有显式忽略该信号,则子进程成为僵尸进程。
下面我们写一个脚本来模拟产生僵尸进程:
#include <stdio.h> #include <sys/types.h> int main() { //fork a child process pid_t pid = fork(); if (pid > 0) //parent process { printf("in parent process, sleep for one miniute...zZ...\n"); sleep(60); printf("after sleeping, and exit!\n"); } else if (pid == 0) { //child process exit, and to be a zombie process printf("in child process, and exit!\n"); exit(0); } return 0; } #生成运行文件 [root@controller ~]# gcc zombie.c -o zombie #运行 [root@controller ~]# ./zombie in parent process, sleep for one miniute...zZ... in child process, and exit! #查看使用 ps 等命令查看系统中僵尸进程,僵尸进程的状态标记为‘Z’: [root@controller ~]# ps aux|grep defunct root 83895 0.0 0.0 0 0 pts/0 Z+ 20:37 0:00 [zombie] <defunct> root 83909 0.0 0.0 112660 972 pts/1 S+ 20:38 0:00 grep --color=auto defunct #如何杀死僵尸进程 [root@controller ~]# kill 83895 [root@controller ~]# ps aux|grep defunct root 83895 0.0 0.0 0 0 pts/0 Z+ 20:42 0:00 [zombie] <defunct> root 83909 0.0 0.0 112660 972 pts/1 S+ 20:42 0:00 grep --color=auto defunct #我们发现如果使用kill 【pid】 无法杀死僵尸进程,可以kill僵尸父进程。父进程死后,僵尸进程成为”孤儿进程”,过继给1号进程init,init始终会负责清理僵尸进程.它产生的所有僵尸进程也跟着消失。 [root@controller ~]# ps -e -o ppid,stat|grep -e '[Zz]'|awk '{print $1}'|xargs kill -9 [root@controller ~]# ./zombie in parent process, sleep for one miniute...zZ... in child process, and exit! Killed
如何避免僵尸进程的产生
处理SIGCHLD信号并不是必须的。但对于某些进程,特别是服务器进程往往在请求到来时生成子进程处理请求。如果父进程不等待子进程结束,子进程将成为僵尸进程(zombie)从而占用系统资源。如果父进程等待子进程结束,将增加父进程的负担,影响服务器进程的并发性能。在Linux下 可以简单地将 SIGCHLD信号的操作设为SIG_IGN。
signal(SIGCHLD,SIG_IGN);这样,内核在子进程结束时不会产生僵尸进程。这一点与BSD4不同,BSD4下必须显式等待子进程结束才能释放僵尸进程或者用两次fork(),而且使紧跟的子进程直接退出,是的孙子进程成为孤儿进程,从而init进程将负责清除这个孤儿进程。
如对您有帮助,支持下呗!
微信
![](https://files.cnblogs.com/files/Dev0ps/WeChat.gif)
支付宝
![](https://files.cnblogs.com/files/Dev0ps/Alipay.gif)