孤儿进程与僵尸进程

来源:互联网 发布:设置首选网络类型4g 编辑:程序博客网 时间:2024/06/04 17:59
一.僵尸进程与孤儿进程定义
僵尸进程:一个子进程在其父进程还没有调用wait()或waitpid()的情况下退出。这个子进程就是僵尸进程。僵尸进程会以终止状态保持在进程表中,并且一直等待父进程都去退出状态代码。

孤儿进程:一个父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程。孤儿进程将被init进程(也就1是1号进程)所收养,并由init进程对它们完成状态收集工作。

    值得注意的是:如果僵尸进程一直未被回收,则会造成空间资源的浪费。



这里是一个三秒钟后子进程变为僵尸状态的例子



写好程序后,在linux中写一个简单监控,监控他们的状态


接下来等待三秒之后,Z状态出现了,这就是我们的僵尸状态


这里显示了其父进程没有退出,也没有wait(),该进程将向父进程发送SIGCHILD信号,进入僵尸状态等待父进程为其收尸。如果父进程一直没有执行wait(),那么该子进程将会持续处于僵尸状态。如果僵尸进程出现太多,有可能出现内存泄漏,这是一个很严重的问题,必须重视。



子进程成为孤儿进程


接下来同样写一个监控去观察他们的状态,结果如下


二、影响:
僵尸进程会占用系统资源,如果很多,则会严重影响服务器的性能(内存泄漏)
孤儿进程不会占用系统资源
处理流程:
只要父进程不等wait()子进程,子进程都将成为孤儿Zombie


if 父进程比子进程先退出
子进程将被init(id = 1)收养,最后的结果是zombie子进程彻底再见,系统资源释放
else
{
子进程的zombie将一直存在,系统资源占用...
if 父进程dead
子进程将被init(id = 1)收养,最后的结果是zombie子进程彻底再见,系统资源释放

else 类似的子进程zombie越来越多,系统就等死了!!!
}
三、如何防止僵尸进程
首先明白如何产生僵尸进程:
1、子进程结束后向父进程发出SIGCHLD信号,父进程默认忽略了它
2、父进程没有调用wait()或waitpid()函数来等待子进程的结束
 捕捉SIGCHLD信号,并在信号处理函数里面调用wait函数

int main(int argc, char **argv){      ...     Signal(SIGCHLD, sig_chld);for(;}     ...}void sig_chld(int signo){      pid_t pid;     int stat;  while ( (pid = waitpid(-1, &stat, WNOHANG)) >; 0)       printf("child %d terminated\n", pid);      return;}


1 0
原创粉丝点击