waitpid函数
只要子进程的状态发生了变化,就会给父进程发信号SIGCHLD。比如子进程暂停执行,恢复执行,这些信号,父进程都可以忽略。
waitpid()
参数pid
pid_t waitpid(pid_t pid, int * status, int options);
pid>0 等待子进程pid
pid=0 waitpid等待和当前调用waitpid一个组的所有子进程
pid=-1 等待所有子进程
pid<-1 表示等 组id=|pid|的所有子进程
参数status
首先,status 分成下面四类:
1 进程正常退出
2 进程被信号终止
3 进程被暂停执行
4 进程被恢复执行
由 4 个宏函数判断到底是哪一种。分别是 WIFEXITED(status)、WIFSIGNALED(status)、WIFSTOPPED(status)和WIFCONTINUED(status)。
waitexited(status)返回true,进程是正常退出,
这时候你可以通过宏 WCOREDUMP(status) 返回 true 还是 false 来判断是否生生了 core 文件。在某些系统中,WCOREDUMP 宏可能没有定义,你需要使用 #ifdef WCOREDUMP ... #endif 来判断。
参数 options 是可个组合选项,这也是参数写 options 而不是 option 原因。
options 的三个可组合选项如下(可以理解成开关组合):
WNOHANG (设置非阻塞,即使子进程全部正常运行,waitpid 也会立即返回 0)
WUNTRACED (可获取子进程暂停状态,也就是可获取 stopped 状态)
WCONTINUED (可获取子进程恢复执行的状态,也就是可获取 continued 状态)
你可以任意使用位或 | 运算符自由组合他们。如果 options 置空(这三个开关都不打开),也就是 0,这意味着 waitpid 函数是阻塞的(在有子进程正常运行的情况下)。
waitpid 能否接收 stopped 和 continued 状态,取决于 WUNTRACED 和 WCONTINUED 开关是否打开。
如果你的 WUNTRACED 开关未打开,waitpid 函数是不会理会子进程停止状态的。同理,如果 WCONTINUED 开关未打开,waitpid 也不会理会子进程恢复执行的状态。后面的实验,读者可以自行验证,代码也会提供。
阻塞信号和未决信号
int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
不仅可以阻塞指定的信号,也可以将之前阻塞的信号撤销,
how参数:SIG_BLOCK选项表示将set参数指示的信号集中的信号添加到进程阻塞集中;
SIG_UNBLOCK选项与功能BLOCK相反,表示将进程阻塞信号集中指定的信号删除
- 未决信号
进程收到信号之后,没有被信号处理函数处理的那些信号,当进程收到信号且该信号被阻塞,首先进入未决信号集中,信号集中的信号被信号处理函数处理,就会从未决信号集中删除