当前位置: 首页 > news >正文

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时生成的结果文件中各个图片及其中指标的含义,帮助大家更深入的理解,以及我们在评估模型时和发表论文时主要关注的参数有那些。本文通过举例训练过程中的某一时间的结果来帮助大家理解&#xf…...

陶建辉在 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 提供的一个函数,用于比较两个值是否相等。它会递归地比较两个对象的属性和值,以判断它们是否相等。 这个函数的作用是: 深度比较对象:递归比较两个对象的每一个属性和嵌套对象的属性,判断它们…...

Ansible常用模块详解(附各模块应用实例和Ansible环境安装部署)

目录 一、ansible概述 1、简介 2、Ansible主要功能: 3、Ansible的另一个特点:所有模块都是幂等性 4、Ansible的优点: 5、Ansible的四大组件: 二、ansible环境部署: 1、环境: 2、安装ansible&#…...

QT中网络编程之发送Http协议的Get和Post请求

文章目录 HTTP协议GET请求POST请求QT中对HTTP协议的处理1.QNetworkAccessManager2.QNetworkRequest3.QNetworkReply QT实现GET请求和POST请求Get请求步骤Post请求步骤 测试结果 使用QT的开发产品最终作为一个客户端来使用,很大的一个功能就是要和后端服务器进行交互…...

Java 并发编程 —— Fork/Join 框架的原理详解

目录 一. 前言 二. 并发和并行 2.1. 并发 2.2. 并行 2.3. 分治法 三. ForkJoin 并行处理框架的理论 3.1. ForkJoin 框架概述 3.2. ForkJoin 框架原理 3.3. 工作窃取算法 四. ForkJoin 并行处理框架的实现 4.1. ForkJoinPool 类 4.2. ForkJoinWorkerThread 类 4.3.…...

3-10岁孩子语文能力培养里程碑

文章目录 基础能力3岁4岁5岁6-7岁(1-2年级)8-9岁(3-4年级)10岁(5年级) 阅读推荐&父母执行3岁4-5岁6-7岁(1-2年级)8-9岁(3-4年级)10岁(5年级&a…...

Vue+ElementUi 基于Tree实现动态节点添加,节点自定义为输入框列

VueElementUi 基于Tree实现动态节点手动添加&#xff0c;节点自定义为输入框列 代码 <el-steps :active"active" finish-status"success" align-center><el-step title"test1"/><el-step title"test2"/><el-st…...

Web前端-JavaScript(js数组和函数)

文章目录 1.数组1.1 数组的概念1.2 创建数组1.3 获取数组中的元素1.4 数组中新增元素1.5 遍历数组 2.函数2.1 函数的概念2.2 函数的使用函数声明调用函数函数的封装 2.3 函数的参数函数参数语法函数形参和实参数量不匹配时 2.4 函数的返回值2.4.1 案例练习 2.5 arguments的使用…...

判断数据是否为整数--函数设计与实现

#定义函数&#xff1a;is_num(s),判断输入的数据是否整数。 #(1)判断是否是数字 def is_num(s):if s.isdigit(): #isdigit()是一个字符串方法&#xff0c;用于检查字符串是否只包含数字字符。如果字符串只包含数字字符&#xff0c;则返回True&#xff1b;否则返回Falsereturn T…...

netty源码:(29)ChannelInboundHandlerAdapter

它实现的方法都有一个ChannelHandlerContext参数&#xff0c;它的方法都是直接调用ChannelHandlerContext参数对应的方法&#xff0c;该方法会调用下一个handler对应的方法。 可以继承这个类&#xff0c;重写感兴趣的方法,比如channelRead. 这个类有个子类&#xff1a;SimpleC…...

Shell脚本应用(二)

一、条件测试操作 Shell环境根据命令执行后的返回状态值〈$?&#xff09;来判断是否执行成功&#xff0c;当返回值为О时表示成功.否则〈非О值)表示失败或异常。使用专门的测试工具---test命令&#xff0c;可以对特定条件进行测试&#xff0e;并根据返回值来判断条件是否成立…...

Kafka基本原理及使用

目录 基本概念 单机版 环境准备 基本命令使用 集群版 消息模型 成员组成 1. Topic&#xff08;主题&#xff09;&#xff1a; 2. Partition&#xff08;分区&#xff09;&#xff1a; 3. Producer&#xff08;生产者&#xff09;&#xff1a; 4. Consumer&#xff08;…...

深圳建网站的专业公司/活动推广宣传方案

PHP 5 及以上版本建议使用以下方式连接 MySQL :MySQLi &#xff1a;MySQLi 只针对 MySQL 数据库&#xff0c;MySQLi 还提供了 API 接口。PDO (PHP Data Objects)&#xff1a;PDO 应用在 12 种不同数据库中。共同点&#xff1a;1. 两者都是面向对象2. 两者都支持预处理语句。 预…...

上海网站建设维护/seo是什么服

Flutter基础—你好&#xff0c;Flutter&#xff01; Flutter基础—开发环境与入门 Flutter基础—第一个Flutter实例 Flutter基础—质感设计 Flutter基础—手势处理 Flutter基础—应用实例 Flutter基础—根据用户输入改变控件 Flutter基础—常用控件之容器 Flutter基础—…...

wordpress怎么提速/百度秒收录排名软件

这么久没有写博客了&#xff0c;今天给大家分享一些多file文件的操作。一般可以用到清理垃圾获取文件大小 删除文件等操作&#xff0c;可以直接用于工具类里面&#xff0c;直接做操作便可以public final class FileUtils { public static long fileLen 0;public static void d…...

平谷区住房城乡建设委官方网站/餐饮营销引流都有什么方法

一、准备工作 Eureka通过运行多个实例&#xff0c;使其更具有高可用性。事实上&#xff0c;这是它默认的熟性&#xff0c;你需要做的就是给对等的实例一个合法的关联serviceurl 二、改造工作 在eureka-server工程中resources文件夹下&#xff0c;创建配置文件application-pe…...

免费门户网站模板/搜狗seo软件

PCB线路板布线有哪些常用规则&#xff1f;文/中信华PCB在我们的日常生活中&#xff0c;我们所接触到的每一种电子设备当中几乎都会出现印刷电路板&#xff0c;如果在某样设备中有电子零件&#xff0c;那么它们也都是镶在大小各异的PCB上。要使电子电路获得最佳性能&#xff0c;…...

网站开发接单平台/竞价培训

作用域闭包递归 &#xff08;自己调自己&#xff09; 简单闭包 function parent() {var x parentfunction son() {var x sonreturn x}return son() } parent() //son闭包写法&#xff1a; /*写法一*/ function parent() {var x parentreturn function son() {var x …...