当一个进程正常或者异常终止后,内核会给父进程发送SIGCHLD信号,因为子进程终止是一个异步事件,所以这个信号也是内核向父进程发送的异步通知,现在需要知道的是调用wait或waitid会发生什么:
- 如果其所有子进程还在运行,则阻塞;
- 如果一个进程已终止,正在等待父进程获取其终止状态,取得该进程的终止状态立即返回。
- 如果它没有任何子进程,返回出错;
#include<sys/wait.h>
pid_t wait(int *status)
pid_t waitid(pid_t pid,int *status,int options)
这两个函数的区别如下:
- 在一个子进程终止前,wait使其调用者阻塞,waitid有一选项,可让调用者不阻塞;
- waitid并不等待在其调用之后的第一个终止子进程,它有若干个选项,可以控制它所等待的进程。
如果子进程已经结束,并且是一个僵死进程,则为他立即返回并取得该子进程的状态,否则wait是其调用者阻塞直到一个子进程终止。如果调用者阻塞,而且他有多个子进程,则在某某一进程终止时,wait就立即返回。
如果一个进程有几个子进程,那么只要有一个子进程终止,wait就返回。如果要等待一个指定的进程终止(如果知道要等待进程的ID),那么该如何做呢?在早期的UNIX版本中,必须调用wait,然后将其返回的进程ID和所期望的进程ID相比较。如果终止进程不是所期望的,则将该进程ID和终止状态保存起来,然后再次调用wait。反复这样做,直到所期望的进程终止。下一次又想等待一个特定进程时,先查看已终止的进程列表,若其中已有要等待的进程,则获取相关信息;否则调用wait。其实,我们需要的是等待一个特定进程的函数。
进程一旦调用了wait,就立即阻塞自己,由wait自动分析是否当前进程的某个子进程已经退出,如果让它找到了这样一个已经变成僵尸的子进程,wait就会收集这个子进程的信息,并把它彻底销毁后返回;如果没有找到这样一个子进程,wait就会一直阻塞在这里,直到有一个出现为止。
参数status用来保存被收集进程退出时的一些状态,它是一个指向int类型的指针。但如果我们对这个子进程是如何死掉的毫不在意,只想把这个僵尸进程消灭掉,(事实上绝大多数情况下,我们都会这样想),我们就可以设定这个参数为NULL,就象下面这样:
pid = wait(NULL);
如果成功,wait会返回被收集的子进程的进程ID,如果调用进程没有子进程,调用就会失败,此时wait返回-1,同时errno被置为ECHILD。
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdlib.h>
main()
{
pid_t pc,pr;
pc=fork()<