Linux(9)——进程(控制篇——下)
目录
三、进程等待
1)进程等待的必要性
2)获取子进程的status
3)进程的等待方法
wait方法
waitpid方法
多进程创建以及等待的代码模型
非阻塞的轮训检测
四、进程程序替换
1)替换原理
2)替换函数
3)函数解释
4)命名理解
三、进程等待
1)进程等待的必要性
- 之前提过子进程退出,父进程如果不读取子进程的退出信息,就可能造成“僵尸进程”的问题,从而造成内存泄漏的问题。
- 再者,一旦子进程进入了僵尸状态,那就连kill -9都杀不亖他,因为没有谁能够杀亖一个死去的进程。
- 最后,父进程创建子进程是要获取子进程的完成任务的情况的。
- 父进程需要通过等待的方式来回收子进程的资源,获取子进程的退出信息。
2)获取子进程的status
下面进程等待使用的两个方法wait方法和waitpid方法都有一个status参数,这是一个输出型参数(输出型参数是函数中用于返回结果或修改调用者变量的参数,通常通过引用或指针实现。如void func(int *output)。),由操作系统进行填写。
如果向status中传递的是NULL,那就表示用户不关心子进程的退出状态。否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程。
status不能简单的当作整形来看待,可以当作位图来看待,具体细节如下图(只研究status低16比特位):
我们从图中可见,status的低16比特位中,高8位表示进程的退出状态,即退出码。当进程被信息杀亖时,则低7位表示终止信息,第8位时core dump标志。
我们可以通一系列的位操作来得出进程的退出码和退出信号。
exitcCode = (status >> 8) & 0xFF; //退出码exitSignal = status & 0x7F;
对于这两个操作,系统提供了两宏来获取退出码以及退出信号。分别是:
- WIFEXITED(status):用于查看是否是正常退出,本质是检查是否收到信号。
- WEXITSTATUS(status):用于获取进程的退出码。
exitNormal = WIFEXITED(status); //是否正常退出exitCode = WEXITSTATUS(status); //获取退出码
敲黑板:
当一个进程是非正常退出的时候,那么该进程的退出码将毫无意义。
3)进程的等待方法
wait方法
函数类型:pid_t wait(int* status);
返回值:成功返回被等待进程pid,失败返回-1。
参数:输出型参数,获取子进程退出状态,不关心则可以设置成为NULL
作用:等待任意子进程
创建子进程后,父进程使用wait方法等待子进程,直到子进程的退出信息被读取,我们可以写个代码验证一下:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> int main()
{ pid_t id = fork(); if(id == 0){ //子进程 int count = 10; while(count--) { printf("我是子进程,PID:%d, PPID:%d\n", getpid(), getppid()); sleep(1); } exit(0); } //父进程 int status = 0; pid_t ret = wait(&status); if(ret > 0){ printf("等待成功...\n"); if(WIFEXITED(status)){ printf("退出码:%d\n", WEXITSTATUS(status)); } } sleep(3); return 0;
}
然后我们可以在开一个会话用来监控进程的状态:
while :; do ps axj | head -1 && ps axj | grep test | grep -v grep;echo "============================================================";sleep 1;done
在下面这图中我们可以看到,当子进程退出,父进程读取到了子进程的退出信息时,子进程就不会变成僵尸状态了。
waitpid方法
函数原型:pid_t waitpid(pid_t pid, int *status, int options);
返回值:
- 当正常返回的时候waitpid返回收集到的子进程的进程ID;
- 如果设置了选项WNOHANG(option),而调用中waitpid发现没有已退出的子进程可收集,则返回0;
- 如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在;
参数:
- pid:当pid=-1,等待任意一个子进程,与wait等效。当pid>0.等待其进程ID与pid相等的子进程。
- status:输出型参数,用来获取子进程的退出状态,不关心可以设置成NULL。
- options:默认为0,表示阻塞等待;当设置为WNOHANG时,若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进程的ID。
返回值:等待任意子进程(可以指定)退出
我们可以写个代码来验证一下,创建子进程后,父进程可以使用waitpid函数一直等待子进程,直到子进程退出后读取子进程的退出信息。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> int main()
{ pid_t id = fork(); if(id == 0){ //子进程 int count = 10; while(count--) { printf("我是子进程,PID:%d, PPID:%d\n", getpid(), getppid()); sleep(1); } exit(0); } //父进程 int status = 0; //pid_t ret = wait(&status); pid_t ret = waitpid(id, &status, 0); if(ret >= 0){ printf("等待成功...\n"); if(WIFEXITED(status)){ printf("退出码:%d\n", WEXITSTATUS(status)); }else{ printf("被信号杀?:%d\n",status & 0x7F); } } sleep(3); return 0;
}
在父进程运行过程中,我们可以使用kill -9命令来将子进程杀亖,这个时候父进程也能成功等待子进程。
敲黑板:
被信号杀亖的进程的退出码是没有意义的。
多进程创建以及等待的代码模型
上面演示的都是父进程的创建以及等待一个子进程,那么接下来我们可以同时创建多个子进程,然后让父进程进程依次等待子进程退出。
下面我们可以同时创建10个子进程,将子进程的pid放到一个id数组中,并将这10个子进程的退出时的退出码设置为该子进程pid对应数组中的下标,之后父进程使用waitpid等待这10个子进程。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h> int main()
{ pid_t ids[10]; for(int i = 0; i < 10; i++){ pid_t id = fork(); if(id == 0){ printf("子进程创建成功:PID:%d\n", getpid()); sleep(3); exit(i); } ids[i] = id; } for(int i = 0; i < 10; i++){ int status = 0; pid_t ret = waitpid(ids[i], &status, 0); if(ret >= 0){ printf("等待子进程成功:PID:%d\n", ids[i]); if(WIFEXITED(status)){ printf("退出码:%d\n", WEXITSTATUS(status)); } else{ printf("被信号杀亖:%d\n", status & 0x7F); } } } return 0;
}
运行完代码,我发现父进程同时创建了了多个子进程,当子进程退出后,父进程再以此读取这些子进程的退出信息。
杀亖进程,父进程依然可以获取子进程的退出信息。
非阻塞的轮训检测
上面的方案,其实是有缺点的,那就是在父进程等待子进程的时候,父进程什么也干不了,这样的等待就是阻塞等待。
而实际上,我们是可以让我们的父进程在等待子进程退出的过程中做一些自己的事情的,这样的等待就是非阻塞等待了。
其实想要实现非阻塞等待也很简单,我们在上面说waitpid时就提过了,在想这个函数第三个参数传入WNOHANG时就可以使得在等待子进程时,如果waitpid函数直接返回0,就不予等待,而等待的子进程若是正常结束,就返回该子进程的pid。
比如,父进程可以选择地调用wait函数,若是等待的子进程还没有退出,那么父进程就可以先去做一些其他的事情了,过一段时间在调用waitpid函数读取子进程的退出信息。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h> int main()
{ pid_t id = fork(); if(id == 0){ int count = 3; while(count--){ printf("子进程在运行:PID:%d, PPID:%d\n", getpid(), getppid()); sleep(3); } exit(0); } while(1){ int status = 0; pid_t ret = waitpid(id, &status, WNOHANG); if(ret > 0){ printf("等待子进程成功。\n"); printf("退出码是:%d\n", WEXITSTATUS(status)); break; } else if(ret == 0){ printf("父进程做了其他事情。\n"); sleep(1); } else{ printf("等待错误。\n"); break; } } return 0;
}
代码的运行结果就是,父进程每隔一段时间就去看看子进程是否退出,如果没就去做自己的事情,知道子进程退出后读取子进程退出的信息。
四、进程程序替换
1)替换原理
用fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往往要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行。调用exec并不创建新进程,所以调用exec前后该进程的id并未改变。
回答两个关于进程程序替换的文题:
问题一:
当进行进程替换时,有没有创建新的进程呢?
进程程序替换之后,该进程的PCB、进程的地址空间以及页表等数据结构都没有发生改变,而只是对原来在物理内存上的代码和数据进行了替换,所有根本没有创建新的进程,而且进程的程序替换前后其pid也是没有改变的。
问题二:
子进程进行进程的程序替换时,是否会影响父进程的代码和数据呢?
不会影响,一开始子进程被父进程创建时代码和数据是和父进程共享的,但是一旦子进程要进行程序替换操作,就意味着子进程需要对代码和数据进行写入操作了,这时就需要对父进程的代码和数据进行拷贝了(写时拷贝),从这里开始子进程和父进程的代码和数据就分离了,所有子进程进行进程的程序替换时不会影响父进程的代码和数据。
2)替换函数
这里的替换函数都是以exec开头的,所以称之为exec函数,总共有六种:
1️⃣int execl(const char *path,const char *arg,...);
相关参数的说明:第一参数是要执行程序的路径,第二个参数是可变参数列表,表示具体如何执行这个程序,同时以NULL结尾。
写一个执行ls程序:
execl("/usr/bin/ls", "ls", "-l", NULL);
2️⃣int execlp(const char *file, const char *arg,...);
相关参数的说明:第一个参数是要执行程序的名字,第二个参数是可变参数列表,表示你要如何执行,也是以NULL结尾。
写一个执行ls程序:
execle("ls", "ls", "-l", NULL);
3️⃣int execle(const char *path, const char *arg,..., char *const envp[]);
相关参数的说明:第一个参数是要执行的程序的路径,第二个参数是可变参数列表,表示如何执行这个程序,也是以NULL结尾。第三个参数是自己设置的环境变量。
这里我们可以设置MYVAL的环境变量,在test中可以使用这个环境变量了。
char* myenp[] = {"MYVAL=2025", NULL};
execle("./test", "test", NULL, myenvp);
4️⃣int execv(const char *path, char *const argv]);
相关参数的说明:第一个参数是要执行程序的路径,第二个参数是一个指针数组,数组中的就是你要如何执行这个程序,数组也是以NULL结尾的。
写一个ls程序:
char* myargv[] = {"ls", "-l", NULL);
execv("/usr/bin/ls", myargv);
5️⃣int execvp(const char *file,char *const argv[]);
相关参数的说明:第一个是要执行的程序的名字,第二个参数是一个指针数组, 数组中的就是你要如何执行这个程序,数组也是以NULL结尾的。
写一个ls程序:
char* myargv[] = {"ls", "-l", NULL};
execvp("ls", myargv);
6️⃣int execve(const char *path, char *const argvl], char *const envp[]);
相关参数的说明: 第一个参数是要执行程序的路径,第二个参数是一个指针数组,数组中的就是你要如何执行这个程序,数组也是以NULL结尾的,第三个参数是自己设置的环境变量。
这里我们可以设置MYVAL的环境变量,在test中可以使用这个环境变量了。
char* myargv[] = {"mycmd", NULL};
char* myenvp[] = {"MYVAL=2025", NULL};
execve("./test", test, myenvp);
3)函数解释
- 这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回。
- 如果调用出错则返回-1
所以exec函数只有出错的返回值而没有成功的返回值。
4)命名理解
其实我们仔细观察就会发现这六个函数都是有规律的,只要掌握了规律是很好记的。
- l(list):表示参数采用列表的形式
- v(vector):参数用数组的形式
- p(path):有p自动搜索环境变量PATH
- e(env):表示自己维护环境变量
函数名 | 参数格式 | 是否带路径 | 是否使用当前环境变量 |
---|---|---|---|
execl | 列表 | 不是 | 是 |
execlp | 列表 | 是 | 是 |
execle | 列表 | 不是 | 不是,须自己组装环境变量 |
execv | 数组 | 不是 | 是 |
execvp | 数组 | 是 | 是 |
execve | 数组 | 不是 | 不是,须自己组装环境变量 |
事实上,我们打开man手册就可以发现execve在man手册第2节其它函数在man手册第3节。也就是 只有execve是真正的系统调用,其它五个函数最终都调用execve。
相关文章:

Linux(9)——进程(控制篇——下)
目录 三、进程等待 1)进程等待的必要性 2)获取子进程的status 3)进程的等待方法 wait方法 waitpid方法 多进程创建以及等待的代码模型 非阻塞的轮训检测 四、进程程序替换 1)替换原理 2)替换函数 3&…...

E. Melody 【CF1026 (Div. 2)】 (求欧拉路径之Hierholzer算法)
E. Melody 思路 将所有出现过的音量和音高看作一个点,一个声音看作一条边,连接起来。那么很容易知道要找的就是图上的一条欧拉路径(类似一笔画问题) 又已知存在欧拉路径的充要条件为:度数为奇数的点的个数为0或者2个…...
@Pushgateway 数据自动清理
文章目录 Pushgateway 数据自动清理一、Pushgateway 数据清理的必要性二、自动清理方案方案1:使用带TTL功能的Pushgateway分支版本方案2:使用Shell脚本定期清理方案3:结合Prometheus记录规则自动清理 三、最佳实践建议四、验证与维护五、示例…...

粽叶飘香时 山水有相逢
粽叶飘香时 山水有相逢 尊敬的广大客户们: 五月初五,艾叶幽香。值此端午佳节,衡益科技全体同仁向您致以最诚挚的祝福! 这一年我们如同协同竞渡的龙舟,在数字化转型的浪潮中默契配合。每一次技术对接、每轮方案优化&a…...

YC-8002型综合变配电监控自动化系统
一 .系统概述 YC-8002型综合变配电监控自动化系统是西安亚川电力科技有限公司为适应广大客户要求,总结多项低 压配电网络自动化工程实例的经验,基于先进的电子技术、计算机和网络通讯等技术自主研发的--套结合本公司网络配电产品的应用于低压配电领域的…...

react diff 算法
diff 算法作为 Virtual DOM 的加速器,其算法的改进优化是 React 整个界面渲染的基础和性能的保障,同时也是 React 源码中最神秘的,最不可思议的部分 diff 算法会帮助我们就算出 VirtualDOM 中真正变化的部分,并只针对该部分进行原…...

近期手上的一个基于Function Grap(类AWS的Lambda)小项目的改造引发的思考
函数式Function是云计算里最近几年流行起来的新的架构和模式,因为它不依赖云主机,非常轻量,按需使用,甚至是免费使用,特别适合哪种数据同步,数据转发,本身不需要保存数据的业务场景,…...
Obsidian 社区插件下载修复
Obsidian 社区插件下载修复 因为某些原因,在国内经常无法下载 Obsidian 的社区插件。这个项目的主要目的就是修复这种情况,让国内的用户也可以无障碍的下载社区插件。 上手指南 下载 obsidian-proxy-github.zip解压 obsidian-proxy-github.zip将解压的…...

VSCode的下载与安装(2025亲测有效)
目录 0 前言1 下载2 安装3 后记 0 前言 丫的,谁懂啊,尝试了各种办法不行的话,我就不得不拿出我的最后绝招了,卸载,重新安装,我经常要重新安装,所以自己写了一个博客,给自己…...

千库/六图素材下载工具
—————【下 载 地 址】——————— 【本章下载一】:https://pan.xunlei.com/s/VORW9TbxC9Lmz8gCynFrgdBzA1?pwdxiut# 【本章下载二】:https://pan.quark.cn/s/829e2a4085d3 【百款黑科技】:https://ucnygalh6wle.feishu.cn/wiki/…...
Ansible模块——Ansible的安装!
Ansible 安装 Ansible 有三种安装方式,源码安装、发行版安装和 Python 安装。 使用发行版安装或 Python 安装两种方式时,Ansible 的安装包有两个,区别如下: • ansible-core:一种极简语言和运行时包,包含…...

差分S参数-信号与电源完整性分析
差分S参数: 由于差分互连中使用差分信号传递信息,接收器最关心的是差分信号的质量,如果互连通道的S参数能直接反映出对差分信号的影响,对分析问题将方便得多。差分互连通道可以看成是一个四端口网络,激励源为单端信号,…...
扣子Coze飞书多维表插件-查询数据
search_record - 查询数据 请求参数 apptoken - 多维表的唯一标识服 可选参数: automatic_fields - 控制是否返回自动计算的字段, true 表示返回。 field_names - 字段名称,用于指定本次查询返回记录中包含的字段。 示例值:["字段1&…...
计算机模拟生物/化学反应有哪些软件?
以下是用于计算机模拟生物/化学反应的软件分类总结,涵盖量子化学、分子动力学、化学信息学及新兴混合方法等方向,结合其核心功能和应用场景进行整理: ⚛️ 一、量子化学计算软件 ChemiQ(本源量子) 类型:量子…...

PostIn V1.1.2版本发布,新增接口评审功能,提升接口质量与合理性
PostIn是一款国产开源免费的接口管理工具,包含项目管理、接口调试、接口文档设计、接口数据MOCK等模块,支持常见的HTTP协议、websocket协议。本周PostIn V1.1.0版本发布,新增接口评审、接口统计功能。 1、版本更新日志 新增 ➢ 接口评审&a…...
MySQL数据归档利器:pt-archiver原理剖析与实战指南
MySQL数据归档利器:pt-archiver原理剖析与实战指南 在MySQL数据库管理中,数据归档是一个永恒的话题——随着业务数据的不断增长,如何高效、安全地将历史数据从生产表迁移到归档表或文件,同时不影响在线业务,是每个DBA和开发者都需要面对的挑战。Percona Toolkit中的pt-ar…...

【论文阅读】User Diverse Preference Modeling by Multimodal Attentive Metric Learning
User Diverse Preference Modeling by Multimodal Attentive Metric Learning 题目翻译:基于多模态注意度量学习的用户不同偏好建模 摘要 提出一个**多模态注意力度量学习(MAML, Multimodal Attentive Metric Learning)**方法,…...
Catch That Cow POJ - 3278
农夫约翰得知了一头逃亡奶牛的位置,想要立即抓住她。他起始于数轴上的点N(0 ≤ N ≤ 100,000),而奶牛位于同一条数轴上的点K(0 ≤ K ≤ 100,000)。农夫约翰有两种移动方式:步行和传送。 * 步行…...

【计算机网络】传输层TCP协议——协议段格式、三次握手四次挥手、超时重传、滑动窗口、流量控制、
🔥个人主页🔥:孤寂大仙V 🌈收录专栏🌈:计算机网络 🌹往期回顾🌹: 【计算机网络】传输层UDP协议 🔖流水不争,争的是滔滔不息 一、TCP协议 UDP&…...

文件服务端加密—minio配置https
1.安装openssl 建议双击安装包默认安装,安装完成之后配置环境变量 如下图则配置成功: 二、生成自签名CA证书及私钥 1、生成私钥 openssl genrsa -out private.key 20482、生成自签名证书 (1)编写openssl.conf文件,…...

界面开发框架DevExpress XAF实践:集成.NET Aspire后如何实现自定义遥测?
DevExpress XAF是一款强大的现代应用程序框架,允许同时开发ASP.NET和WinForms。DevExpress XAF采用模块化设计,开发人员可以选择内建模块,也可以自行创建,从而以更快的速度和比开发人员当前更强有力的方式创建应用程序。 .NET As…...

多卡训练核心技术详解
多卡训练核心技术详解 多卡训练 主要围绕分布式环境初始化、模型并行化、数据分片和梯度同步展开。下面结合您的代码,详细解释这些核心部分: 并行执行命令 torchrun --nproc_per_node=5 TokenLossMulCard.py 1. 分布式环境初始化 def init_distributed():init_process_…...

深度学习全面掌握指南
一、引言 深度学习是机器学习的一个分支,它通过构建多层神经网络来模拟人类大脑的信息处理方式,从而实现对复杂数据的自动特征提取和模式识别。近年来,深度学习在计算机视觉、自然语言处理、语音识别等领域取得了巨大的突破,引发…...

magic-api配置Git插件教程
一、配置gitee.com 1,生成rsa密钥,在你的电脑右键使用管理员身份运行(命令提示符),执行下面命令 ssh-keygen -t rsa -b 2048 -m PEM一直按回车键,不需要输入内容 找到 你电脑中的~/.ssh/id_rsa.pub 文件…...
算法打卡第11天
36.有效的括号 (力扣20题) 示例 1: **输入:**s “()” **输出:**true 示例 2: **输入:**s “()[]{}” **输出:**true 示例 3: **输入:**s “(]”…...
【决策分析】基于Excel的多变量敏感性分析解决方案
在Excel中实现多变量敏感性分析(3个及以上变量)需要结合更灵活的工具和方法,因为Excel内置的数据表功能仅支持最多双变量分析。以下是针对多变量场景的解决方案,按复杂度和实用性排序: 方法1:场景管理器 摘…...
php:5.6-apache Docker镜像中安装 gd mysqli 库 【亲测可用】
Dockerfile 代码如下: FROM php:5.6-apache# 使用Debian归档源 RUN echo "deb http://archive.debian.org/debian stretch main contrib non-free" > /etc/apt/sources.list && \echo "deb http://archive.debian.org/debian-security s…...
小程序跳转H5或者其他小程序
1. h5跳转小程序有两种情况 (1)从普通浏览器打开的h5页面跳转小程序使用wx-open-launch-weapp可以实现h5跳转小程序 <wx-open-launch-weappstyle"display:block;"v-elseid"launch-btn":username"wechatYsAppid":path…...

【AI赋能,视界升级】智微智能S134 AI OPS,重构智慧大屏未来
智慧教室中,教师通过电子白板,4K高清课件、3D教学模型同步呈现,后排学生也能看清画面细节,课堂变得趣味十足;智能会议室里,会议内容、多人云会议多屏投放依旧畅通清晰,会议纪要自动生成Word/PPT…...

外网访问可视化工具 Grafana (Linux版本)
Grafana 是一款强大的可视化监控指标的展示工具,可以将不同的数据源数据以图形化的方式展示,不仅通用而且非常美观。它支持多种数据源,如 prometheus 等,也可以通过插件和 API 进行扩展以满足各种需求。 本文将详细介绍如何在本地…...