Linux之进程(五)(进程控制)
目录
一、进程创建
1、fork函数创建进程
2、fork函数的返回值
3、fork常规用法
4、fork调用失败的原因
二、进程终止
1、进程终止的方式
2、进程退出码
3、进程的退出方法
三、进程等待
1、进程等待的必要性
2、wait函数
3、waitpid函数
四、进程程序替换
1、概念
2、原理
3、进程替换函数
一、进程创建
在之前的学习中,我们已经简单使用了fork函数创建一个进程。下面我们来具体讲一讲fork创建进程。
1、fork函数创建进程
在linux中fork函数是一个非常重要的函数,它从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程。
返回值:子进程中返回0,父进程返回子进程pid,创建进程出错则返回-1。
上面这些我们非常熟悉了。接着我们来看看下面的内容。
进程调用fork,当控制转移到内核中的fork代码后,内核做:
1、分配新的内存块和内核数据结构给子进程
2、将父进程部分数据结构内容拷贝至子进程
3、添加子进程到系统进程列表当中
4、fork返回,开始调度器调度
fork之后,父子进程代码共享(即fork之前的代码由父进程独立执行,而fork之后的代码父子两个执行流分别执行)。并且,fork之后,父进程和子进程谁先执行完全由调度器决定。
因为代码是只读的,所以代码可以完全共享。虽然代码是父子共享的,但是,有的数据必须分离,要各自都有一份。于是我们就要用到写时拷贝的技术,而这我们已经在上一节进程地址空间中讲到了。
那么,为什么我们不在创建进程的时候,就直接拷贝分离呢?下面我们来具体讲一讲。
我们创建子进程的目的是为了让它去帮助我们完成和父进程不一样的任务,而不一样的任务所需要用到的数据可能是不一样的。而操作系统无法提前知道子进程需要什么数据来完成任务。比如:有的数据是只读的,所以父子进程完全可以共享,而有的数据又可写,为了不让父进程数据的修改影响子进程或者子进程数据的修改影响父进程,我们要写时拷贝。
简单来说,如果全部拷贝,那么我们可能会将一些子进程不会用到的数据拷贝一份,这样就浪费空间,而且效率低下。即使要用到,但是只是读取,完全可以父子进程共享数据,避免空间中有两份一模一样的数据。
所以将来会被父或子进程修改的数据值得拷贝。但即使是操作系统也无法预先知道谁会被访问修改,所以要用写时拷贝技术。

2、fork函数的返回值
~ fork函数为什么有两个返回值
fork之后,有两个执行流,父子进程代码是共享的,所以return会被调度两次,被父子进程各自执行return的。
~ 为什么父进程返回子进程pid,给子进程返回0
父亲只有一个,孩子可以有多个,孩子找父亲具有唯一性。所以给父进程返回子进程pid便于标识子进程。
3、fork常规用法
1、一个父进程希望复制自己,使父子进程同时执行不同的代码段。例如,父进程等待客户端请求,生成子进程来处理请求。
 2、一个进程要执行一个不同的程序。例如子进程从fork返回后,调用exec函数。
4、fork调用失败的原因
fork有时也会调用失败,原因一般有两个:1、系统中有太多的进程 2、实际用户的进程数超过了限制。
二、进程终止
1、进程终止的方式
1、代码跑完,结果正确
2、代码跑完,结果错误
3、代码没有跑完,程序崩溃
2、进程退出码
首先我们来看一个问题:我们在编写代码时,在main函数的结尾,我们总是 return 0。那么main函数返回值的意义是什么呢?为什么总是返回0呢?返回其他值行不行呢?
实际上main函数只是用户级别代码的入口,main函数也是被其他函数调用的,所以当main函数调用结束后就应该给操作系统返回相应的退出信息,而这个所谓的退出信息就是以退出码的形式作为main函数的返回值返回的。
一般以0表示代码成功执行完,以非0表示代码执行过程中出现了错误。比如下面的代码:

代码运行完后,我们可以使用 echo $? 命令查看最近一个进程的退出码

因为在程序中,运行成功了那就是成功了,而如果失败了,那么失败的原因有很多种。所以用0表示成功,非0的不同值可以表示不同的错误。因此我们可以通过退出码来定位程序的错误原因。
我们使用strerror函数可以通过错误码,获取该错误码在C语言当中对应的错误信息:


注:程序崩溃的时候,退出码没有意义。
3、进程的退出方法
正常终止
~ return 返回:return只有在main函数中表示直接终止进程,并返回退出码,(在其他函数中表示返回值)。
~ exit函数:是一个库函数。在代码的任何地方调用,都表示直接终止进程。并且exit函数在退出进程前会做一系列工作:
1、执行用户通过atexit或on_exit定义的清理函数。
 2、关闭所有打开的流,所有的缓存数据均被写入。
 3、调用_exit函数终止进程。
如下面的代码:因为exit是库函数,exit终止进程前会将缓冲区当中的数据输出。


~ _exit:是系统调用函数。_exit函数也可以在代码中的任何地方退出进程,但是_exit函数会直接终止进程,并不会在退出进程前会做任何收尾工作。
如下面的代码:缓冲区当中的数据将不会被输出。


~ exit 和 _exit 的区别
从下面的图中,我们可以知道:exit会刷新缓冲区的数据,而_exit不会刷新。

异常终止
1、使用 ctrl+c或者kill -9使进程异常退出。
2、代码错误导致进程运行时异常退出。例如:代码当中存在野指针问题使得进程运行时异常退出,或是出现除0情况使得进程运行时异常退出等。
三、进程等待
1、进程等待的必要性
1、子进程退出,父进程如果不读取子进程的退出信息,子进程就会变成僵尸进程,进而造成内存泄漏。
2、进程一旦变成僵尸状态,那么即使是kill -9 也无能为力,因为谁也没有办法杀死一个已经死去的进程。
3、因为子进程是由父进程创建出来帮助父进程完成任务的,所以进程完成任务后,父进程有必要知道子进程将任务完成得怎么样了。
4、父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息。
2、wait函数
原型:pid_t wait(int* status) 可以等待任意子进程,等待成功返回等待子进程pid,失败返回-1。status:输出型参数,获取子进程退出状态,不关心则可以设置成为NULL。

当子进程退出后,父进程读取了子进程的退出信息,子进程也就不会变成僵尸进程了。

3、waitpid函数
原型:pid_t waitpid(pid_t pid, int* status, int options) 等待指定子进程或任意子进程。
返回值:
 1、等待成功返回被等待进程的pid。
 2、如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0。
 3、如果调用中出错,则返回-1,这时errno会被设置成相应的值以指示错误所在。
参数:pid:Pid=-1,等待任一个子进程。与wait等效。Pid>0,等待其进程ID与pid相等的子进程。
status:该参数是一个输出型参数,由操作系统填充 ,如果传递NULL,表示不关心子进程的退出状态信息。否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程。
options: 若pid指定的子进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该子进程的ID。


获取子进程status
status就是将子进程的退出信息传递给父进程的重要参数。但是它不能简单的当作整形来看待。因为它的32个比特位中,只有次低8位表示进程退出码信息。如下图:

因此我们只能通过下面的代码来获取子进程的正确退出码: 

获取子进程的终止信号
status参数的低7位,表示子进程的终止信号,我们可以通过下面的代码获得。
status & 0x7f除了上面的我们可以自己写代码进行退出码的获取,我们还可以使用提供的宏进行退出码的获取:
WIFEXITED(status): 若为正常终止子进程返回的状态,则为真(非0)。(查看进程是否是正常退出)
 WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的退出码) 
父进程拿到子进程的退出码
既然进程之间是相互独立的,而且退出码也还是子进程的数据,那么为什么父进程可以通过wait和waitpid函数,拿到子进程的退出信息,它是怎么拿到的呢?
子进程变成僵尸进程后,虽然子进程已经退出了,但是子进程的PCB任然会保留下来,子进程的task_struct 里面保留了该子进程退出时的退出结果信息。所以父进程可以通过wait和waitpid函数读取子进程task_struct 中的退出结果信息(拿到子进程的退出码和退出信号)。
options
默认为0,代表父进程阻塞等待,WNOHANG代表父进程非阻塞等待。
potions传入WNOHANG,等待的子进程若是没有结束,那么waitpid函数将直接返回0,不予以等待。而等待的子进程若是正常结束,则返回该子进程的pid。
阻塞等待:当子进程未退出时,父进程都在一直等待子进程退出,在等待期间,父进程不能做任何事情,这种等待叫做阻塞等待。
非阻塞等待:父进程没有一直等待子进程退出,而是当子进程未退出时父进程可以做一些自己的事情,当子进程退出时再读取子进程的退出信息,这种等待就叫非阻塞等待。
四、进程程序替换
1、概念
我们知道,在父进程使用fork函数创建了一个子进程后,父子进程共享代码。也就是说子进程使用的是父进程的代码。那么如果子进程要执行和父进程不一样的代码,该怎么办呢?这就需要使用进程的程序替换了。
进程程序替换,顾名思义就是使用一个新的程序替换原有的程序,进程将执行新程序的代码,而不再执行原有程序的代码。(通过特定的接口,加载磁盘上的一个全新的程序,加载到调用进程的地址空间中)
2、原理

如上图所示,将磁盘上的 test2.exe 加载到内存,并和 test1.exe的页表重新建立映射关系,这时进程的代码和数据就变成了 test2.exe的,这就是进程替换。进程替换可以通过系统调用接口实现。
3、进程替换函数
execl 函数
函数原型:int execl(const char *path, const char *arg, ...)。我们知道,运行一个程序的前提是要先找到该程序,所以该函数的第一个参数是程序的路径。第二个参数是可变参数列表,表示你要如何执行这个程序,并以NULL结尾。
execlp函数
函数原型:int execlp(const char *file, const char *arg, ...) 。第一个参数是要执行程序的名字,第二个参数是可变参数列表,表示你要如何执行这个程序,并以NULL结尾。
execle函数
函数原型:int execle(const char *path, const char *arg, ..., char *const envp[])。第一个参数是要执行程序的路径,第二个参数是可变参数列表,表示你要如何执行这个程序,并以NULL结尾,第三个参数是你自己设置的环境变量。
execv函数
函数原型:int execv(const char *path, char *const argv[])。第一个参数是要执行程序的路径,第二个参数是一个指针数组,数组当中的内容表示你要如何执行这个程序,数组以NULL结尾。
execvp函数
函数原型:int execvp(const char *file, char *const argv[])。第一个参数是要执行程序的名字,第二个参数是一个指针数组,数组当中的内容表示你要如何执行这个程序,数组以NULL结尾。
execve函数
int execve(const char *path, char *const argv[], char *const envp[])。第一个参数是要执行程序的路径,第二个参数是一个指针数组,数组当中的内容表示你要如何执行这个程序,数组以NULL结尾,第三个参数是你自己设置的环境变量。
相关文章:
 
Linux之进程(五)(进程控制)
目录 一、进程创建 1、fork函数创建进程 2、fork函数的返回值 3、fork常规用法 4、fork调用失败的原因 二、进程终止 1、进程终止的方式 2、进程退出码 3、进程的退出方法 三、进程等待 1、进程等待的必要性 2、wait函数 3、waitpid函数 四、进程程序替换 1、概念…...
 
63. 不同路径 II 23.12.21(二)
一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” )。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为 “Finish”)。 现在考虑网格中有障碍物。那么从左上角到右下角…...
 
【线性代数】两个向量组等价,其中一个向量组线性无关,另一个向量组也是线性无关吗?
一、问题 两个向量组等价,其中一个向量组线性无关,另一个向量组也是线性无关吗? 二、答案 不一定,当两个向量组中的向量个数也相同时,结论才成立.若向量个数不相同,结论不成立. 例如: 向量组一:(1,0),(0,1) 向量组二:(1,0),(0,1),(1,1) 两…...
 
c语言:指针作为参数传递
探究实参与形参它们相互独立 由于主调函数的变量a,b与被调函数的形参x,y它们相互独立。函数 swap 可以修改变量x,y,但是却无法影响到主调函数中的a,b。 现在利用取地址运算符,分别打印它们的首地址&#x…...
 
YOLOv5性能评估指标->mAP、Precision、Recall、FPS、Confienc (讲解论文关注的主要指标)
简介 这篇博客,主要给大家讲解我们在训练yolov5时生成的结果文件中各个图片及其中指标的含义,帮助大家更深入的理解,以及我们在评估模型时和发表论文时主要关注的参数有那些。本文通过举例训练过程中的某一时间的结果来帮助大家理解…...
 
陶建辉在 CIAS 2023 谈“新能源汽车的数字化”
近年,中国的新能源汽车发展迅猛,在全球竞争中表现出色,已经连续 8 年保持全球销量第一。在新兴技术的推动下,新能源汽车的数字化转型也正在加速进行,从汽车制造到能源利用、人机交互,各个环节都在进行数字化…...
 
PSP - 结构生物学中的机器学习 (NIPS MLSB Workshop 2023.12)
欢迎关注我的CSDN:https://spike.blog.csdn.net/ 本文地址:https://spike.blog.csdn.net/article/details/135120094 Machine Learning in Structural Biology (机器学习在结构生物学中) 网址:https://www.mlsb.io/ Workshop at the 37th Co…...
 
某领先的集成电路研发中心:建立跨网交换平台 杜绝数据泄露风险
1、客户介绍 某技术领先的集成电路研发中心,是产学研合作的国家级集成电路研发中心,致力于解决重大共性技术的研发及服务支撑问题。该中心积极探索国际化道路,不断提升国际影响力,与多家国际著名集成电路企业和研发机构建立技术合…...
 
map|动态规划|单调栈|LeetCode975:奇偶跳
作者推荐 【贪心算法】【中位贪心】.执行操作使频率分数最大 涉及知识点 单调栈 动态规划 map 题目 给定一个整数数组 A,你可以从某一起始索引出发,跳跃一定次数。在你跳跃的过程中,第 1、3、5… 次跳跃称为奇数跳跃,而第 2、…...
 
从安全性角度,看“可信数字底座”有何价值
文章目录 每日一句正能量前言概念对比安全技术对比思考与建议 每日一句正能量 不管现在有多么艰辛,我们也要做个生活的舞者。 前言 万向区块链此前提出“可信数字底座”这一概念和技术,即将区块链与物联网、人工智能、隐私计算等数字化技术相融合&#…...
 
软件设计模式:UML类图
文章目录 前言一、📖设计模式概述1.软件设计模式的产生背景2.软件设计模式3.设计模式分类 二、📣UML图1.类图概述2.类的表示法3.类与类之间的关系关联关系(1)单向关联(2)双向关联(3)…...
力扣题目学习笔记(OC + Swift)15. 三数之和
15. 三数之和 给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k ,同时还满足 nums[i] nums[j] nums[k] 0 。请 你返回所有和为 0 且不重复的三元组。 注意:答案中不可以包含重复的三元…...
 
想将电脑屏幕共享到iPhone上,但电脑是Linux系统,可行吗?
常见Windows系统或macOS系统的电脑投屏到手机,难道Linux系统的电脑要投屏就是个难题吗? 想要将Linux系统投屏到iPhone、iPad、安卓设备、鸿蒙设备,其实你可以利用软件AirDroid Cast和Chrome浏览器!连接同一网络就可以直接投屏。 第…...
 
大华 DSS 城市安防数字监控系统 SQL 注入漏洞
漏洞简介 大华DSS数字监控系统itcBulletin接口对传入的数据没有预编译和充足的校验,导致该接口存在SQL注入漏洞,可通过注入漏洞获取数据库敏感信息。 资产测绘 app“dahua-DSS” 漏洞复现 POC: POST /portal/services/itcBulletin?wsdl HTTP/1.1 H…...
 
vue中的侦听器和组件之间的通信
目录 一、侦听器 监听基本数据类型: 监听引用数据类型: 计算属性和watch区别? 二、组件通信/传值方式 1.父子组件传值 父组件给子组件传值: (1)props (2)provide inject &…...
maven-shade-plugin有什么用
maven-shade-plugin 是 Maven 的一个插件,用于创建可执行的 JAR 文件,并且可以将所有依赖项打包到一个 JAR 文件中。 该插件的主要用途是创建包含所有依赖项的“fat” JAR(也称为“uber” JAR),使得应用程序可以作为一…...
本地部署 OpenVoice
本地部署 OpenVoice OpenVoice 介绍Qwen-Audio Github 地址部署 OpenVoice克隆代码库创建虚拟环境使用 pip 安装 pytorch使用 pip 安装依赖下载 checkpoint运行 Web UI OpenVoice 介绍 通过 MyShell 进行即时语音克隆。 Qwen-Audio Github 地址 https://github.com/myshell-…...
 
【模式识别】解锁降维奥秘:深度剖析PCA人脸识别技术
🌈个人主页:Sarapines Programmer🔥 系列专栏:《模式之谜 | 数据奇迹解码》⏰诗赋清音:云生高巅梦远游, 星光点缀碧海愁。 山川深邃情难晤, 剑气凌云志自修。 目录 🌌1 初识模式识…...
 
大模型赋能“AI+电商”,景联文科技提供高质量电商场景数据
据新闻报道,阿里巴巴旗下淘天集团和国际数字商业集团都已建立完整的AI团队。 淘天集团已经推出模特图智能生成、官方客服机器人、万相台无界版等AI工具,训练出了自己的大模型产品 “星辰”; 阿里国际商业集团已成立AI Business,…...
深度比较(lodash 的 isEqual 方法)
_.isEqual() 是 Lodash 提供的一个函数,用于比较两个值是否相等。它会递归地比较两个对象的属性和值,以判断它们是否相等。 这个函数的作用是: 深度比较对象:递归比较两个对象的每一个属性和嵌套对象的属性,判断它们…...
Cursor实现用excel数据填充word模版的方法
cursor主页:https://www.cursor.com/ 任务目标:把excel格式的数据里的单元格,按照某一个固定模版填充到word中 文章目录 注意事项逐步生成程序1. 确定格式2. 调试程序 注意事项 直接给一个excel文件和最终呈现的word文件的示例,…...
QMC5883L的驱动
简介 本篇文章的代码已经上传到了github上面,开源代码 作为一个电子罗盘模块,我们可以通过I2C从中获取偏航角yaw,相对于六轴陀螺仪的yaw,qmc5883l几乎不会零飘并且成本较低。 参考资料 QMC5883L磁场传感器驱动 QMC5883L磁力计…...
 
【网络安全产品大调研系列】2. 体验漏洞扫描
前言 2023 年漏洞扫描服务市场规模预计为 3.06(十亿美元)。漏洞扫描服务市场行业预计将从 2024 年的 3.48(十亿美元)增长到 2032 年的 9.54(十亿美元)。预测期内漏洞扫描服务市场 CAGR(增长率&…...
生成 Git SSH 证书
🔑 1. 生成 SSH 密钥对 在终端(Windows 使用 Git Bash,Mac/Linux 使用 Terminal)执行命令: ssh-keygen -t rsa -b 4096 -C "your_emailexample.com" 参数说明: -t rsa&#x…...
C++.OpenGL (10/64)基础光照(Basic Lighting)
基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...
Spring Boot+Neo4j知识图谱实战:3步搭建智能关系网络!
一、引言 在数据驱动的背景下,知识图谱凭借其高效的信息组织能力,正逐步成为各行业应用的关键技术。本文聚焦 Spring Boot与Neo4j图数据库的技术结合,探讨知识图谱开发的实现细节,帮助读者掌握该技术栈在实际项目中的落地方法。 …...
 
Linux --进程控制
本文从以下五个方面来初步认识进程控制: 目录 进程创建 进程终止 进程等待 进程替换 模拟实现一个微型shell 进程创建 在Linux系统中我们可以在一个进程使用系统调用fork()来创建子进程,创建出来的进程就是子进程,原来的进程为父进程。…...
 
OPENCV形态学基础之二腐蚀
一.腐蚀的原理 (图1) 数学表达式:dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy) 腐蚀也是图像形态学的基本功能之一,腐蚀跟膨胀属于反向操作,膨胀是把图像图像变大,而腐蚀就是把图像变小。腐蚀后的图像变小变暗淡。 腐蚀…...
 
保姆级教程:在无网络无显卡的Windows电脑的vscode本地部署deepseek
文章目录 1 前言2 部署流程2.1 准备工作2.2 Ollama2.2.1 使用有网络的电脑下载Ollama2.2.2 安装Ollama(有网络的电脑)2.2.3 安装Ollama(无网络的电脑)2.2.4 安装验证2.2.5 修改大模型安装位置2.2.6 下载Deepseek模型 2.3 将deepse…...
 
Python Ovito统计金刚石结构数量
大家好,我是小马老师。 本文介绍python ovito方法统计金刚石结构的方法。 Ovito Identify diamond structure命令可以识别和统计金刚石结构,但是无法直接输出结构的变化情况。 本文使用python调用ovito包的方法,可以持续统计各步的金刚石结构,具体代码如下: from ovito…...
