为何要fork()两次来避免产生僵尸进程??

最近安装书上说的,开始搞多进程了..看到了一个好帖子,学习学习

http://blog.sina.com.cn/s/blog_9f1496990100y420.html

 

首先我们要明白,为什么要避免僵尸进程----->进程是一种资源,无用的进程在内存里面无谓的耗费资源.

当我们只fork()一次后,存在父进程和子进程。这时有两种方法来避免产生僵尸进程

  • 父进程调用waitpid()等函数来接收子进程退出状态(这种情况下,子进程结束后成为僵尸进程,等待父进程回收..问题是如果父进程一直while()循环执行不到waitpid呢,此时子进程这个僵尸进程就会很长久的,甚至是伴随软件周期来存在,这种情况必须要重视!)
  • 父进程先结束,子进程则自动托管到Init进程(pid = 1)(这种情况下,子进程结束后会被Init自动回收)

目前先考虑子进程先于父进程结束的情况:

  • 若父进程未处理子进程退出状态,在父进程退出前,子进程一直处于僵尸进程状态。
  • 若父进程调用waitpid()(这里使用阻塞调用确保子进程先于父进程结束)来等待子进程结束,将会使父进程在调用waitpid()后进入睡眠状态,只有子进程结束父进程的waitpid()才会返回。 如果存在子进程结束,但父进程还未执行到waitpid()的情况,那么这段时期子进程也将处于僵尸进程状态。

由此,可以看出父进程与子进程有父子关系,除非保证父进程先于子进程结束或者保证父进程在子进程结束前执行waitpid(),子进程均有机会成为僵尸进程。那么如何使父进程更方便地创建不会成为僵尸进程的子进程呢?这就要用两次fork()了。

父进程一次fork()后产生一个子进程随后立即父进程执行waitpid(子进程pid, NULL, 0)来等待子进程结束,然后子进程fork()后产生孙子进程随后立即exit(0)。这样子进程顺利终止(父进程仅仅给子进程收尸,并不需要子进程的返回值),然后父进程继续执行。这时的孙子进程由于失去了它的父进程(即是父进程的子进程),将被转交给Init进程托管。于是父进程与孙子进程无继承关系了,它们的父进程均为Init,Init进程在其孙子进程结束时会自动收尸,这样也就不会产生僵尸进程了。

posted @ 2016-05-03 09:17  哈哈不是嘎嘎  阅读(919)  评论(1编辑  收藏  举报