等待函数
/* 如果子进程已经终止,并且是一个僵死进程,wait立即返回并取得该子进程状态。 否则wait使其调用者阻塞,直当一个子进程终止。 如果有多个子进程,在其中一个子进程终止,wait就立即返回 */ pid_t wait(int *status); //一个子进程终止前,wait使其调用者阻塞。 pid_t waitpid(pid_t pid,int *status,int options); //附加选项,可控制它所等待的进程。
参数:
pid : 参数pid为欲等待的子进程识别码,其具体含义如下:
参数值 |
说明 |
pid<-1 |
等待进程组号为pid绝对值的任何子进程。 |
pid=-1 |
等待任何子进程,此时的waitpid()函数就退化成了普通的wait()函数。 |
pid=0 |
等待进程组号与目前进程相同的任何子进程,也就是说任何和调用waitpid()函数的进程在同一个进程组的进程。 |
pid>0 |
等待进程号为pid的子进程。 |
status : 如果status不是一个空指针,则终止进程的终止状态就存放它所指的单元内。
如果不关心终止状态,则可将该参数设为空指针.
Linux提供了一些非常有用的宏来帮助解析这个状态信息,这些宏都定义在sys/wait.h头文件中。主要有以下几个:
宏 |
说明 |
WIFEXITED(status) |
如果子进程正常结束,它就返回真;否则返回假。 |
WEXITSTATUS(status) |
如果WIFEXITED(status)为真,则可以用该宏取得子进程exit()返回的结束代码。 |
WIFSIGNALED(status) |
如果子进程因为一个未捕获的信号而终止,它就返回真;否则返回假。 |
WTERMSIG(status) |
如果WIFSIGNALED(status)为真,则可以用该宏获得导致子进程终止的信号代码。 |
WIFSTOPPED(status) |
如果当前子进程 被暂停了,则返回真;否则返回假。 |
WSTOPSIG(status) |
如果WIFSTOPPED(status)为真,则可以使用该宏获得导致子进程暂停的信号代码。 |
WCOREDUMP(status) |
如果子进程导致内核转存,返回真。只有WIFSIGNALED返回真时,才应该使用。并非所有平台都支持这个宏,使用时应放在#ifdef WCOREDUMP ... #endif内部。 当子进程接收到停止信号时true |
WIFCONTINUED(status) |
如果子进程通过SIGCONT恢复则返回一个非零值。 |
options : 提供了一些另外的选项来控制waitpid()函数的行为。如果不想使用这些选项,则可以把这个参数设为0。
其具体含义如下:
主要使用的有以下两个选项:
参数 |
说明 |
WNOHANG |
如果pid指定的子进程没有结束,则waitpid()函数立即返回0,而不是阻塞在这个函数上等待;如果结束了,则返回该子进程的进程号。 |
WUNTRACED |
如果有处于停止状态的进程将导致调用返回。 |
WCONTINUED |
如果停止了的进程由于SIGCONT信号的到来而继续运行,调用将返回。 |
这些参数可以用“|”运算符连接起来使用。
返回值:
若 成功 , 返回 终止子进程的ID
若 失败 ,返回 -1
如果指定WNOHANG标志,同时pid指定的进程状态没有发生变化,将返回0。
失败原因:
- 系统中已经有了太多的进程。
- 该实际用户ID的进程
p int waitid(idtype_t idtype, id_t id, siginfo_t *infop, int options);
//类似waitpid,但提供了更多的灵活性
返回值:
若 成功执行 , 返回 0。
若 WNOHANG标志被设置而id指定的子进程状态没有发生变化, 返回 0。
若 失败 ,返回 -1。
参数:
idtype,id : 这两个参数结合在一起指出应选择等待哪些子进程,可能情况有 :
idtype == P_PID //等待 一 特定进程。id包含要等待的子进程的进程ID
idtype == P_PGID //等待一特定进程组中的任一子进程,id包含要等待子进程的进程组ID
idtype == P_ALL //等待任一子进程,忽略id
options : 指定的感兴趣的子进程状态改变标志有以下常量,可以通过or运算加以组合 :
前3个常量之一必须指定
WEXITED //等待已终止的子进程。
WSTOPPED //等待由于信号已停止执行的子进程。
WCONTINUED //等待由于信号已恢复执行的子进程。
WNOHANG //如无可用的子进程退出状态,立即返回而非阻塞。( 作用如同waitpid)
WNOWAIT //保留子进程的可等待状态,后面的wait调用可再次获取子进程的状态信息。
infop : 成功执行返回时,waitid将填充infop所指向结构体的如下字段:
si_pid //子进程标识。
si_uid //子进程的真实用户标识。
si_signo //总被设置为SIGCHLD。
si_status //子进程的退出状态,或者导致子进程退出,停止执行或恢复执行的信号,需要根据si_code字段来解释。
si_code //可能值有CLD_EXITED(子进程调用_exit退出),
CLD_KILLED(子进程被信号杀死),
CLD_STOPPED(信号导致子进程停止执行),
CLD_CONTINUED(信号恢复子进程继续执行)。
失败原因:
ECHILD:参数指定的进程不存在,或者并非调用进程的子进程
EINTR ;WNOHANG不被设置,同时捕捉到一个不被阻塞的信号或SIGCHLD信号
EINVAL :options参数无效
pid_t wait3(int *statloc, int options, struct rusage *rusage);
// 等待所有的子进程;
pid_t wait4(pid_t pid, int *statloc, int options, struct rusage *rusage);
// 可以像waitpid一样通过pid指定要等待的子进程
//这两个函数提供的而wait、 waitid和waitpid函数没有提供的唯一的特性是一个额外的参数,来允许内核返回终止的进程和它的子进程使用的资源的汇总。
返回值:
若 成功 , 返回 终止子进程的ID
若 出错 ,返回 -1
参数:
option:
参数 |
说明 |
WNOHANG |
如果pid指定的子进程没有结束,则waitpid()函数立即返回0,而不是阻塞在这个函数上等待;如果结束了,则返回该子进程的进程号。 |
WUNTRACED |
如果有处于停止状态的进程将导致调用返回。 |
WCONTINUED |
如果停止了的进程由于SIGCONT信号的到来而继续运行,调用将返回。 |
rusage:
通过ru指针可以返回子进程的资源使用情况。
struct rusage { struct timeval ru_utime; struct timeval ru_stime; long ru_maxrss; long ru_ixrss; long ru_idrss; long ru_isrss; long ru_minflt; long ru_majflt; long ru_nswap; long ru_inblock; long ru_oublock; long ru_msgsnd; long ru_msgrcv; long ru_nsignals; long ru_nvcsw; long ru_nivcsw; };
也可以通过getrusage函数获取进程资源使用情况。
int getrusage ( int who, struct rusage *ru );
who可取RUSAGE_SELF、RUSAGE_CHILDREN,分别获取当前进程的资源使用情况和所有已终止且被父进程获取其终止状态的所有子进程的资源使用总情况。