孤儿进程
一个父进程退出,而它的一个或多个子进程还在运行,那么这些子进程将成为孤儿进程。
孤儿进程将被 init进程(进程号为1)所收养,并由 init 进程对它们完成状态收集工作。
僵尸进程
一个进程使用 fork 创建子进程,如果子进程退出,而父进程并没有调用 wait 或者 waitpid 获取子进程的状态信息,那么子进程的进程描述符仍然保存在系统中。这种进程称为僵尸进程。
僵尸进程是一个进程必然会经过的过程:这是每个子进程在结束时都要经过的阶段。
如果子进程在 exit() 之后,父进程没有来得及处理,这时用 ps 命令就能看到子进程的状态时 “Z”。如果父进程能及时处理,可能用 ps 命令就来不及看到子进程的僵尸状态,但这并不等于子进程不经过僵尸状态。
如果父进程在子进程结束之前退出,则子进程将由 init 接管。init 将会以父进程的身份对僵尸进程进行处理。
危害:
如果进程不调用 wait/waitpid 的话,那么保留的字段就不会释放,其进程号会一直被占用,但是系统所能使用的进程号是有限的,如果大量产生僵尸进程,将因为没有可用的进程号而导致系统不能产生新的进程。
外部消灭:
通过 kill 发送 SIGTERM 或者 SIGKILL 信号消灭产生的僵尸进程,它产生的僵尸进程就变成了孤儿进程,这些孤儿进程会被 init 进程接管,init 进程会 wait() 这些孤儿进程,释放它们占用的系统进程表中的资源。
内部解决:
- 子进程退出时向父进程发送 SIGCHILD 信号,父进程处理 SIGCHILD 信号。在信号处理函数中调用 wait 进行处理僵尸进程。
- fork 两次,原理是将子进程变成孤儿进程,从而其父进程变成 init 线程,通过 init 进程可以处理僵尸进程。