系统编程-进程-wait、waitpid和WIFEXITED系列宏 超级详解
1. wait、waitpid 函数简介

补充:对于waitpid,如果返回值为0,表示指定去等待的子进程尚未结束。

该系列宏的使用方法展示:

PART1
实验思路:
使用wait系统调用让父进程给子进程收尸,并获取子进程的返回值。
同时,正常终止。
直接打印获取的返回值会与实际返回的值不一样,进而引出检查WIFEXITED/WEXITSTATUS(status)等系列宏。
实验1: waitpid或waitpid,使用展示, 同时,直接使用printf打印出子进程的返回值
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h> int main(void)
{
int status = 0;
pid_t pid;
printf("pid: %ld\n", (long)getpid()); pid = fork();
if (pid < 0) {
printf("fork error");
}
else if (pid == 0) {
sleep(2);
printf("pid: %ld, ppid: %ld\n", (long)getpid(), (long)getppid()); exit(3);
} if (waitpid(pid, &status, 0) != pid) { // if (wait(&status) != pid) { 这里也可以使用wait, 我这里对waitpid的使用也是阻塞方式的
printf("waitpid error");
}
else{
printf("status =%d\n", status);
} return 0;
}
编译运行:

可以看到,本实验中的waitpid在2秒后才返回,成功替子进程收尸。
但是我们打印的status不对劲,我们的代码内子进程返回值是3, 打印的768是什么鬼? 于是我们做实验2,使用WIFEXITED/WEXITSTATUS(status)等系列宏。
实验2: 在实验1的基础上新增使用WIFEXITED/WEXITSTATUS(status)等系列宏
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h> void out_status(int status){ if(WIFEXITED(status)){
printf("normnal exit: %d \n", WEXITSTATUS(status)); }else if(WIFSIGNALED(status)){
printf("abnormal term: %d \n", WTERMSIG(status)); }else if(WIFSTOPPED(status)){
printf("stopped sig: %d \n", WSTOPSIG(status)); }else{
printf("unknow sig");
}
} int main(void)
{
int status = 0;
pid_t pid;
printf("pid: %ld\n", (long)getpid()); pid = fork();
if (pid < 0) {
printf("fork error");
}
else if (pid == 0) {
sleep(2);
printf("pid: %ld, ppid: %ld\n", (long)getpid(), (long)getppid()); exit(3);
} if (wait(&status) != pid) {
printf("waitpid error");
}
else{
printf("status =%d\n", status);
out_status(status);
} return 0;
}
编译运行:

到此为止,wait和waitpid的阻塞式的使用,以及子进程正常返回的实验,顺利完成。
我们还需要判断子进程的异常终止、和判断子进程是否暂停了, 所以我们接着来完善这部分的知识体系。
PART2
子进程异常终止 - 实验:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h> void out_status(int status){ if(WIFEXITED(status)){
printf("normnal exit: %d \n", WEXITSTATUS(status)); }else if(WIFSIGNALED(status)){
printf("abnormal term: %d \n", WTERMSIG(status)); }else if(WIFSTOPPED(status)){
printf("stopped sig: %d \n", WSTOPSIG(status)); }else{
printf("unknow sig");
}
} int main(void)
{
int status = 0;
pid_t pid;
printf("pid: %ld\n", (long)getpid()); pid = fork();
if (pid < 0) {
printf("fork error");
}
else if (pid == 0) {
sleep(2);
printf("pid: %ld, ppid: %ld\n", (long)getpid(), (long)getppid()); // 我们可以在子进程内新增几行除零运算的代码,来引发子进程异常终止
int i=89, j=0;
int k = i/j;
printf("k=%d \n", k); exit(3);
} if (wait(&status) != pid) {
printf("waitpid error");
}
else{
printf("status =%d\n", status);
out_status(status);
} return 0;
}
编译运行,同时查看下8号信号是什么信号


由度娘知,SIGFPE是系统发出的针对进程内进行非法运算的一个信号。
至此,我们又多了一个技能,可以判断子进程是否被异常终止,以及具体的异常终止原因了!
PART3
判断子进程是否暂停了
要判断子进程是否暂停了,不能使用wait,只能使用waitpid,而且需要加上特定的参数选项。

重申下列知识点:

由此可见,
waitpid函数的最后一个参数配置为WUNTRACED,则后续判断子进程返回状态时,需要结合WIFSTOPED宏、
waitpid函数的最后一个参数配置为WCONTINUED,则后续判断子进程返回状态时,需要结合WIFCONTINUED宏。
实验环节
实验目的:判断子进程是否暂停了、是否又继续运行起来了
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h> void out_status(int status){ if(WIFEXITED(status)){
printf("normnal exit: %d \n", WEXITSTATUS(status)); }else if(WIFSIGNALED(status)){
printf("abnormal term: %d \n", WTERMSIG(status)); }else if(WIFSTOPPED(status)){
printf("stopped sig: %d \n", WSTOPSIG(status)); }else if(WIFCONTINUED(status)){
printf("WIFCONTINUED \n");
}else{
printf("unknow sig \n");
}
} int main(void)
{
int status = 0;
pid_t pid;
printf("pid: %ld\n", (long)getpid()); pid = fork();
if (pid < 0) {
printf("fork error");
}
else if (pid == 0) {
int i=0;
printf("--child process begins running, pid=%d\n", getpid());
while(1){
i++;
if(i == 1000000)
i=0;
}
} pid = waitpid(pid, &status, WUNTRACED);
printf("pid=%d \n", pid);
out_status(status); pid = waitpid(pid, &status, WCONTINUED);
printf("pid=%d \n", pid);
out_status(status); return 0;
}
运行:
一个终端内:

另一个终端内:

分析:
先向子进程发送19号信号,会使得父进程解除阻塞,获取到子进程被暂停了这一信息。
之后父进程又再次调用阻塞接口,等待子进程继续运行,当在该终端内发送18号信号,即可让子进程继续运行,从而使得父进程解除阻塞,获得子进程又继续运行了这一信息。
补充:
19) SIGSTOP 20) SIGTSTP
19号信号和29号信号的相同点: 都可以使得进程暂停,并且收到SIGCONT信号后可以让进程重新运行。
19号信号和29号信号的不同点: SIGSTOP不可以捕获(即使用信号处理函数)。
.
系统编程-进程-wait、waitpid和WIFEXITED系列宏 超级详解的更多相关文章
- Linux系统编程@进程通信(一)
进程间通信概述 需要进程通信的原因: 数据传输 资源共享 通知事件 进程控制 Linux进程间通信(IPC)发展由来 Unix进程间通信 基于System V进程间通信(System V:UNIX系统 ...
- linux系统编程-进程
进程 现实生活中 在很多的场景中的事情都是同时进行的,比如开车的时候 手和脚共同来驾驶汽车,再比如唱歌跳舞也是同时进行的: 如下是一段视频,迈克杰克逊的一段视频: http://v.youku.com ...
- nginx高性能WEB服务器系列之四配置文件详解
nginx系列友情链接:nginx高性能WEB服务器系列之一简介及安装https://www.cnblogs.com/maxtgood/p/9597596.htmlnginx高性能WEB服务器系列之二 ...
- mongo 3.4分片集群系列之五:详解平衡器
这个系列大致想跟大家分享以下篇章: 1.mongo 3.4分片集群系列之一:浅谈分片集群 2.mongo 3.4分片集群系列之二:搭建分片集群--哈希分片 3.mongo 3.4分片集群系列之三:搭建 ...
- mongo 3.4分片集群系列之六:详解配置数据库
这个系列大致想跟大家分享以下篇章: 1.mongo 3.4分片集群系列之一:浅谈分片集群 2.mongo 3.4分片集群系列之二:搭建分片集群--哈希分片 3.mongo 3.4分片集群系列之三:搭建 ...
- legend3---Windows 7/8/10 系统下Laravel框架的开发环境安装及部署详解(Vagrant + Homestead)
legend3---Windows 7/8/10 系统下Laravel框架的开发环境安装及部署详解(Vagrant + Homestead) 一.总结 一句话总结: 1.安装的话就是下载好git,va ...
- Hexo系列(二) 配置文件详解
Hexo 是一款优秀的博客框架,在使用 Hexo 搭建一个属于自己的博客网站后,我们还需要对其进行配置,使得 Hexo 更能满足自己的需求 这里所说的配置文件,是位于站点根目录下的 _config.y ...
- css3系列之transform详解translate
translate translate这个参数的,是transform 身上的,那么它有什么用呢? 其实他的作用很简单,就是平移,参考自己的位置来平移 translate() translateX() ...
- Spring Boot 2.x基础教程:进程内缓存的使用与Cache注解详解
随着时间的积累,应用的使用用户不断增加,数据规模也越来越大,往往数据库查询操作会成为影响用户使用体验的瓶颈,此时使用缓存往往是解决这一问题非常好的手段之一.Spring 3开始提供了强大的基于注解的缓 ...
- ThreeJS系列1_CinematicCameraJS插件详解
ThreeJS系列1_CinematicCameraJS插件详解 接着上篇 ThreeJS系列1_CinematicCameraJS插件介绍 看属性的来龙去脉 看方法作用 通过调整属性查看效果 总结 ...
随机推荐
- [oeasy]python0074_设置高亮色_color_highlight_ansi_控制终端颜色
更多颜色 回忆上次内容 上次我们搞的还是颜色 FG foreground 前景色 30-37 BG background 背景色 40-47 这些 都可以和字体样式 结合起来 难道 就这几种颜色 吗? ...
- c语言的编译与调试
1. GCC/G++ gcc和g++是GNU Compiler Collection中的编译器,分别用于编译C和C++程序.它们的编译过程主要包括四个步骤:预处理(Pre-Processing).编译 ...
- 14、Spring之基于注解的声明式事务
14.1.概述 14.1.1.编程式事务 事务功能的相关操作全部通过自己编写代码来实现: Connection conn = ...; try { // 开启事务:关闭事务的自动提交 conn.set ...
- 【Spring Data JPA】04 JPQL和原生SQL
@Transactional注解 让Spring处理事务 不需要自己每次都手动开启提交回滚 FINDONE & GETONE的区别? findone是立即加载 getone是延迟加载,配合事务 ...
- Windows11重置后出现Windows.old文件夹无法删除,报错C:\Windows.old\WINDOWS\System32\WDI - 目录不是空的。Win11系统Windows.old能删除吗?Windows.old怎么删
问题: Windows11重置后出现Windows.old文件夹无法删除,报错C:\Windows.old\WINDOWS\System32\WDI - 目录不是空的. 网上的各种方法均不奏效: ht ...
- 读论文《基于 GA - BP 的汽车行李箱盖内板冲压成形工艺优化》 —— 如何使用AI技术优化模具产业中工件冲压工艺
最近到了模具公司工作,本来以为身边同事对模具生产和工件生产的流程(大致流程)会比较了解,结果一问才知道基本都是一问三不知,大家都在模具公司工作但是貌似很多人干的和模具生产和工件制造的工作关联性并不强, ...
- 如何配置域名的 CNAME —— 添加记录集时,为什么会提示“与已有解析记录冲突”?
参考: https://support.huaweicloud.com/dns_faq/dns_faq_016.html https://developer.qiniu.com/fusion/kb/1 ...
- EDI企业订单报文系统——冷链物流管理系统——低代码的应用
参考: 驳"低代码开发取代程序员"论 为什么专业开发者也需要低代码? =========================================== 推荐视频: https ...
- C# 委托和闭包
前言 本文只是为了复习,巩固,和方便查阅,一些知识点的详细知识会通过相关链接和文献标记出来. 委托是什么 大部分的解释是 委托是一个对方法的引用,可以不用自己执行,而是转交给其他对象.就好比每天都有一 ...
- 曝光!Apache SeaTunnel Catalog 功能设计为何能大大简化用户启用步骤?
Catalog(目录)提供了关于数据库.表格和访问数据所需的信息的元数据,以及统一的 API 来管理元数据,验证连接,让元数据对 Sources(数据源).Sinks(数据汇)和 Web 可访问. C ...