Linux wait与waitpid

1. 函数原型:

#include <sys/wait.h>

pid_t wait(int *statloc);

pid_t waitpid(pid_t pid, int *statloc, int options);

ret=成功返回进程ID或者0,失败返回-1

(1) statloc是一个整型指针;如果statloc不是一个空指针,则终止进程的终止状态就存放在它指向的单元内;如果不关心终止状态,则可将该参数指定为空指针;

(2) waitpid中的pid参数作用如下:

     pid==-1 等待任一子进程。这个waitpid与wait等效; wait(xx) == waitpid(-1, xx, 0);

     pid>0 等待进程ID与pid相等的子进程;

     pid==0 等待其组ID等于调用进程ID的任一子进程;

     pid < -1 等待期组ID等于pid绝对值的任一子进程;

(3) waitpid的options常量:

     0-不使用任何选项;

     WCONTINUED-若支持作业控制,那么由pid指定的任一子进程在暂停后已经继续,但其状态未被报告,则返回其状态;

     WNOHANG-若pid指定的子进程并不是立即可用的,则waitpid不阻塞,此时其返回值为0

     WUNTRACED-若某实现支持作业控制,而由pid指定的任一子进程已处于暂停状态,但是其状态自暂停以后还未报告过,则返回其状态;WIFSTOPPED宏确定返回值是否对应一个暂停子进程;

(4) 检查终止状态的宏:

WIFEXITED(status)-若为正常终止子进程返回的状态,则为真;对于这种情况可执行WEXITSTATUS(status),取子进程传送给exit, _exit, _Exit参数的低八位;

WIFSIGNALED(status)-若为异常终止子进程返回的状态,则为真(接到一个不捕捉的信号)。对于这种情况,可执行WTERMSIG(status),获取子进程终止的信号编号;另外有些实现定义宏WCOREDUMP(status),若以产生终止进程的core文件,则返回真;

WIFSTOPPED(status)-若为当前暂停进程的返回状态,则为真;对于这种情况,可执行WSTOPSIG(status),获取使子进程暂停的信号编号;

WIFCONTINUED(status)-若在作业控制暂停后,已经继续的子进程返回了状态,则为真;(仅用于waitpid)

2. wait与waitpid区别:

(1) 在一个进程终止前,wait使其调用者阻塞,而waitpid有一个选项,可以使调用者不阻塞;

(2) waitpid并不等待其调用后的第一个终止子进程,它有若干个选项,可以控制它所等待的子进程;

waitpid提供了wait函数没有的三个功能:

(1) waitpid可以等待一个特定的进程,而wait则返回任一终止子进程的状态;

(2) waitpid提供了一个wait的非阻塞版本。有时用户希望获取一个子进程的状态,但不想阻塞;

(3) waitpid支持作业控制;

3. SIGCHLD信号处理多个子进程:

正确做法:

1 void sig_child(int signo)
2 {
3     pid_t pid;
4     int stat;
5 
6     while ((pid = waitpid(-1, &stat, WNOHANG)) > 0)
7         printf("child %d terminated
", pid);
8     return;
9 }

在多个子进程可能同时终止的时候,同时产生多个SIGCHLD信号,因为信号不排队,所以信号处理函数可能只被调用一次,从而有多个子进程没有wait成为僵尸进程;

所以我们需要在产生信号的时候,对所有结束的子进程进行回收,于是使用while循环;

wait不能使用在while中,因为wait是阻塞的,比如当某子进程一直不停止,它将一直阻塞,从而影响了父进程;

于是使用waitpid,options设置为WNOHANG,可以在非阻塞情况下处理多个退出子进程;

原文地址:https://www.cnblogs.com/wanpengcoder/p/5310983.html