Linux 进程终止 进程等待

目录

进程终止

退出码

错误码

代码异常终止(信号详解)

exit

_exit

进程等待

概念

等待的原因

wait

函数原型

参数

返回值

 监控脚本

waitpid

概念

函数原型

 参数

返回值

WIFEXITED(status)

WEXITSTATUS(status)

问题

为什么不用全局变量获得子进程的退出信息?

父进程如何得知子进程的退出信息?

模拟非阻塞等待轮询逻辑

重点


进程终止

退出码

  • 退出码是程序在结束时返回给操作系统的一个状态码,表示程序执行的最终结果,退出码通常用于告知操作系统或调用进程该程序是否成功执行
  • echo $?:Shell自动维护的特殊变量,可以打印出最后一个子进程执行完毕后的退出码(echo是内建命令)

错误码

for(int i = 0; i < 134; i++)printf("%d: %s\n", i, strerror(i));
  •  errno是一个全局变量(通过errno.h引入),用于存储最近一次系统调用库函数执行失败时的错误代码,错误时会将错误码设置到errno中
  • 可以通过strerror将错误码转换为错误描述

代码异常终止(信号详解)

  • 出异常时不看错误码,因为异常机制本身已经传递了足够的信息来描述错误情况
  • 程序崩溃:进程调度中出现异常,异常信息会被OS检测到,OS通过发送信号的方式杀掉、终止(释放)对应的进程
  • 异常通常是程序员自己写出来的
  • 一个进程是否出现异常,看有无收到信号
  • 父进程通过  信号数字  和  退出码  来判断子进程任务完成的怎样

exit

  • exit(int status):status 进程的退出码
  • 等价于在main函数中直接return,任意地点调exit,表示进程退出,后序代码不执行,直接终止进程
  • exit时候,会将冲刷缓冲区
  • exit 函数的底层封装了 _exit 系统调用接口

_exit

  • 是一个系统调用接口,直接与操作系统内核交互
  • _exit时候,不会冲刷缓冲区,内核之上是系统调用接口,说明缓冲区(C语言提供的)不在内核中
  • ★是一个底层的、直接与内核交互的系统调用,它的目标是立即终止进程,而不是进行任何用户空间的清理工作

进程等待

概念

  • 通过wait/waitpid,让父进程对子进程进行资源回收的等待过程

等待的原因

  • 解决子进程的僵尸问题带来的内存泄露
  • 子进程要将父进程给的任务的完成结果(即,子进程的退出信息->进程的退出码,信号编号)返回给父进程
  • 简:OS回收子进程资源获得进程的退出信息

wait

函数原型

#include <sys/wait.h>pid_t wait(int *status);

参数

  • 输出型参数
  • status:用于存储子进程的退出状态。父进程可以通过此参数判断子进程是正常退出还是因异常终止
  • 如果异常了,退出码就没用了;实测:子进程中途异常,exit code 为0

返回值

  • 成功时返回已终止子进程的进程 ID(pid_t 类型)
  • 如果没有子进程或调用失败,返回 -1,并设置 errno
void worker()
{int cnt = 3;while(cnt--){printf("child process, pid:%d, ppid%d\n", getpid(), getppid());sleep(1);}
}void creatSubProcess()
{pid_t id = fork();if(id == 0) {worker();exit(0);}else{    sleep(6);    pid_t rid = wait(NULL);if(rid == id)    {                                                                                    printf("wait success:pid: %d, rid: %d\n", getpid(), rid);    }    sleep(3);    }   
}

 监控脚本

  • 父进程结束的时候,会自动回收子进程,但是也可以在父进程运行中途回收
  • pid_t wait(int *status); status:指向整数的指针,用于存储子进程的退出状态信息
  • wait是一个系统调用接口;声明,相关的宏和数据类型在 sys/wait.h
  • wait能够回收处于僵尸状态的子进程;如果执行到父进程执行到wait这行代码时,子进程没退出,父进程会在wait这行代码上阻塞等待,知道子进程运行完毕,wait会对其回收;是一种等待软件资源的阻塞,一个进程 等待另一个进程执行完成
  • 僵尸进程无法被信号杀掉,只能通过父进程回收

waitpid

概念

  • waitpid 是一种系统调用,使父进程可以等待特定子进程的终止,或按指定的条件等待所有子进程终止,waitpid 通过 pid 参数指定要等待的进程,提供了灵活的子进程管理方式

函数原型

#include <sys/wait.h>pid_t waitpid(pid_t pid, int *status, int options);

 参数

  • pid:指定要等待的子进程
    • pid > 0:等待进程 ID 等于 pid 的子进程。
    • pid = 0:等待与调用进程在同一进程组的任一子进程
    • pid = -1:等待任一子进程(等价于 wait)
    • pid < -1:等待进程组 ID 等于 |pid| 的任一子进程

  • status:用于存储子进程的退出状态
  • options:控制等待行为
    • 0:阻塞等待子进程退出
    • WNOHANG非阻塞模式如果没有已退出的子进程,立即返回
    • WUNTRACED:如果子进程暂停(收到 SIGSTOP 等信号),则返回该子进程状态。
    • WCONTINUED:如果子进程被恢复运行(收到 SIGCONT 信号),则返回该子进程状态。

返回值

  • 成功时返回已终止的子进程的 pid。
  • 如果 WNOHANG 选项被设置,且没有已退出的子进程,返回 0。
  • 出错时返回 -1,并设置 errno

WIFEXITED(status)

  • #define WIFEXITED(status)   (((status) & 0x7F) == 0)
  • W:wait  IF:是否  EXITED:退出
  • 一个宏,用于检查一个子进程是否正常退出,返回非零值时,表示子进程通过 exitreturn 语句正常退出

WEXITSTATUS(status)

  • #define WEXITSTATUS(status)   (((status) >> 8) & 0xFF)
  • W:wait  EXIT:退出  STATUS:状态  
  • wait/waitpid的方式来等待的子进程的退出状态    其由退出码呈现
  • 用于获取子进程的退出状态码,只有在 WIFEXITED(status) 返回非零时(其实就是1),才能使用 WEXITSTATUS(status) 宏来获取子进程的退出码

问题

为什么不用全局变量获得子进程的退出信息?

  • 写实拷贝...进程间是相互独立的父进程无法直接拿到子进程的数据,只能通过系统调用获得
  • 像是一种通信

父进程如何得知子进程的退出信息?

  • 父进程调用系统调用接口来获得子进程的退出信息
  • 1.父进程给waitpid传status变量的地址 2.子进程结束后,代码的退出信息会回写到task_struct中,(task_struct中有这些字段,救就为了给父进程一个交代,也方面父进程去拿)
  • 之后再按照一定的方式回写给status  3.将子进程的exit_state改为X(X肯定是一个宏)
  • 我觉得回收资源是最重要的一点,这样可以创建更多的子进程帮我们完成任务;其次就是任务完成的结果
  • 父进程等待子进程的过程中,被链入子进程的等待队列;task_struct 本身不直接包含等待队列(wait_queue)成员

模拟非阻塞等待轮询逻辑

#include <stdio.h>    
#include <unistd.h>    
#include <stdlib.h>    
#include <sys/types.h>    
#include <sys/wait.h>    #define MAX_WORKER 5    
typedef void (*work)();    void father_work1()    
{    printf("father doing work1\n");    
}    void father_work2()    
{    printf("father doing work2\n");    
}    void father_work3()    
{    printf("father doing work3\n");                                                                                                                                               
}    
void worker()    
{    int cnt = 3;    while(cnt--)    {    printf("child process, pid_t: %d, ppid_t: %d, cnt: %d\n", getpid(), getppid(), cnt);    sleep(1);    }    
}    
void initArray(work array[])    
{    for(size_t i = 0; i < MAX_WORKER; i++)    array[i] = NULL; 
}void addWork(work array[], work w)
{for(size_t i = 0; i < MAX_WORKER; i++){                                                                                                                                                                             if(array[i] == NULL){array[i] = w;break;}}return;
}void doingWork(work array[])
{for(size_t i = 0; i < MAX_WORKER; i++)if(array[i]) array[i](); //(*array[i])();也可以//*array[i]这只是解引用函数,() 这才是调用函数
}
int main()
{work array[MAX_WORKER]; initArray(array);addWork(array, father_work1);addWork(array, father_work2);addWork(array, father_work3);pid_t id = fork();if(id == 0){worker();exit(1);}else{while(1){pid_t rid = waitpid(id, NULL, WNOHANG);if(rid > 0){printf("wait success\n");break;}else if(rid == 0){printf("father doing other thing\n");doingWork(array);}else{printf("wait failed\n");break;}sleep(1);}}return 0;
}
  • 代码风格是一个不错的点,尤其是对array的初始化和填值
  • 代码的核心目的就是简单的模拟:父进程不用一直等待没有被回收的子进程
  • waitpid直接可以返回,父进程去做别的工作,过一段时间再来检查子进程是否可以被回收

重点

进程终止

  1. 退出码,信号数字;错误码
  2. exit与_exit的区别

进程等待

  1. 进程等待的概念、原因
  2. wait与waitpid
  3. WIFEXITED与WEXITSTAUTS

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/58876.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[MySQL]DQL语句(一)

查询语句是数据库操作中最为重要的一系列语法。查询关键字有 select、where、group、having、order by、imit。其中imit是MySQL的方言&#xff0c;只在MySQL适用。 数据库查询又分单表查询和多表查询&#xff0c;这里讲一下单表查询。 基础查询 # 查询指定列 SELECT * FROM …

Java 批量导出Word模板生成ZIP文件到浏览器默认下载位置

是不是你们要找的&#xff01;是不是你们要找的&#xff01;是不是你们要找的&#xff01; 先看效果&#xff1a; 1.word模板格式 2.模板位置 3.需要的依赖 <!--POI-TL实现数据导出到word模板--><dependency><groupId>cn.hutool</groupId><artifact…

Flarum:简洁而强大的开源论坛软件

Flarum简介 Flarum是一款开源论坛软件&#xff0c;以其简洁、快速和易用性而闻名。它继承了esoTalk和FluxBB的优良传统&#xff0c;旨在提供一个不复杂、不臃肿的论坛体验。Flarum的核心优势在于&#xff1a; 快速、简单&#xff1a; Flarum使用PHP构建&#xff0c;易于部署&…

CPU用户时间百分比

在计算机系统中&#xff0c;"CPU用户时间百分比&#xff08;CPU User Time&#xff09;"是一个性能监控指标&#xff0c;它描述了CPU在用户模式下执行的累积时间与总的CPU时间的比例。这个指标可以帮助我们了解系统在执行用户态程序时的负载情况。下面是一些关于CPU用…

ubuntu20.04安装ros与rosdep

目录 前置配置 配置apt清华源 配置ros软件源 添加ros安装源&#xff08;中科大软件源&#xff09; 设置秘钥 更新源 ros安装 安装ros 初始化 rosdep 更新 rosdep 设置环境变量 安装 rosinstall 安装验证 启动海龟仿真器 操控海龟仿真器 rosdep安装更新 安装 使用…

mobile频段要查找、设置并获取相关参数,该怎么破?

今天我们一起来学习查找和设置mobile频段&#xff0c;并获取相关参数。 一、mobile概述 1.1 简介 “4G mobile”指的是第四代移动通信技术&#xff0c;常用于描述通过4G网络进行的高速无线数据传输和通信。4G网络最显著的特征是其高速数据传输能力。理论上&#xff0c;4G可以…

「C/C++」C++11 之<thread>多线程编程

✨博客主页何曾参静谧的博客📌文章专栏「C/C++」C/C++程序设计📚全部专栏「VS」Visual Studio「C/C++」C/C++程序设计「UG/NX」BlockUI集合「Win」Windows程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「PK」Parasolid函数说明目…

[SAP ABAP] SMW0上传模板

通常来说&#xff0c;一个批量导入的程序必须使用指定的模板&#xff0c;我们需要将模板保存到SAP系统中&#xff0c;以便用户下载并更改。这里我们可以使用事务码SMW0解决上述的问题 1.选择二进制类型 2.输入存放的包 3.创建对象 选择需要进行上传的本地模板文件到SAP系统中 …

LSTM模型改进实现多步预测未来30天销售额

项目源码获取方式见文章末尾&#xff01; 600多个深度学习项目资料&#xff0c;快来加入社群一起学习吧。 《------往期经典推荐------》 项目名称 1.【BiLSTM模型实现电力数据预测】 2.【卫星图像道路检测DeepLabV3Plus模型】 3.【GAN模型实现二次元头像生成】 4.【CNN模型实…

中科蓝汛GPIO操作说明

第一种写法&#xff1a; GPIO配置输入模式 //内部上拉 GPIOBDE | BIT(4); //数字IO使能: 0为模拟IO, 1 为数字IO GPIOBDIR | BIT(4); //控制IO的方向: 0为输出, 1为输入. GPIOBFEN & ~BIT(4);//0:当作通用GPIO使用 //1:当作其它功能性IO GPIOBPU | BIT(4); //10K上拉…

3DDFA-V3——基于人脸分割几何信息指导下的三维人脸重建

1. 研究背景 从二维图像中重建三维人脸是计算机视觉研究的一项关键任务。在虚拟现实、医疗美容、计算机生成图像等领域中&#xff0c;研究人员通常依赖三维可变形模型&#xff08;3DMM&#xff09;进行人脸重建&#xff0c;以定位面部特征和捕捉表情。然而&#xff0c;现有的方…

高并发编程

一台64G内存的服务器QPS可以达到9W&#xff0c;TPS&#xff08;事务&#xff09;可以达到5K&#xff0c;每个TPS大约包含18个QPS.只读的话QPS可以达到30~40万.阿里云有相关测试工具、测试方法、测试结果。 1、volatile 保证可见性&#xff0c;禁止指令重排&#xff0c;避免多线…

HTML 基础标签——表单标签<form>

文章目录 1. `<form>` 标签:定义表单容器2. `<input>` 标签:多用途输入控件3. `<textarea>` 标签:多行文本输入框4. `<select>` 标签:下拉选择框5. `<option>` 标签:下拉菜单选项6. `<button>` 标签:按钮元素7. `<label>` 标签…

GraphQL 与 Elasticsearch 相遇:使用 Hasura DDN 构建可扩展、支持 AI 的应用程序

作者&#xff1a;来自 Elastic Praveen Durairaju GraphQL 提供了一种高效且灵活的数据查询方式。本博客将解释 Hasura DDN 如何与 Elasticsearch 配合使用&#xff0c;以实现高性能和元数据驱动的数据访问。 此示例的代码和设置可在此 GitHub 存储库 - elasticsearch-subgraph…

智能座舱相关术语全解及多模态交互在智能座舱中的应用

文章目录 座舱相关术语全解1. 智能座舱2. UFS3. 多模态交互4. 3D虚拟引擎5. AR/VR6. GNSS7. TTS8. DPU9. 摄像头10. 屏幕/显示器11. 音频12. 无线连接13. 其他组件 多模态交互在智能座舱中有以下一些应用 座舱相关术语全解 1. 智能座舱 智能座舱&#xff08;intelligent cabi…

selinux和防火墙

目录 selinux 1、selinux的说明 2、selinux的工作原理 安全上下文四个字段&#xff1a; 访问过程&#xff1a; 3、selinux的启动、关闭与查看 &#xff08;1&#xff09;SELinux三种模式 &#xff08;2&#xff09;修改安全上下文 4、selinux对linux服务的影响 防火墙 …

制氮机分子筛的材质选择

制氮机分子筛的材质选择对于其性能和效率至关重要。作为制氮设备中的核心部件&#xff0c;分子筛承担着将空气中的氮气与氧气有效分离的重任。以下是对制氮机分子筛常用材质的详细探讨&#xff1a; 制氮机分子筛的主要材质 碳分子筛(CMS) 碳分子筛由活性炭经过特殊工艺加工而成…

【运动的&足球】足球场地区域图像分割系统源码&数据集全套:改进yolo11-RFAConv

改进yolo11-ContextGuidedDown等200全套创新点大全&#xff1a;足球场地区域图像分割系统源码&#xff06;数据集全套 1.图片效果展示 项目来源 人工智能促进会 2024.10.28 注意&#xff1a;由于项目一直在更新迭代&#xff0c;上面“1.图片效果展示”和“2.视频效果展示”展…

Java keytool keystore的相关概念介绍与使用

参考资料 Java证书工具keytool用法总结keytool-简介CA证书介绍与格式转换密钥&#xff0c;私钥&#xff0c;公钥的区分とほほのOpenSSL入門 目录 一. 概念1.1 keystore文件1.2 keytool1.3 非对称加密1.3.1 加密解密1.3.2 数字签名认证1.3.3 通信中的加密与签名过程 1.4 证书格…

MFC图形函数学习04——画矩形函数

MFC中绘制矩形函数是MFC的基本绘图函数&#xff0c;它的大小和位置由左上角和右下角的坐标决定&#xff1b;若想绘制的矩形边框线型、线宽、颜色以及填充颜色都还需要其它函数的配合。 一、绘制矩形函数 原型&#xff1a;BOOL Rectangle(int x1,int y1,int x2,int y2); …