[Linux][进程控制][进程程序替换]详细解读

目录

  • 1.进程创建
    • 1.fork函数初识
    • 2.fork函数返回值
    • 3.写时拷贝
    • 4.fork之后,父子进程代码共享
    • 5.fork常规用法
    • 6.fork调用失败的原因
  • 2.进程终止
    • 0.进程终止时,操作系统做了什么?
    • 1.进程退出场景
    • 2.进程常见退出方法
    • 4 _exit函数(系统接口)
    • 4.exit函数(库函数)
  • 3.进程等待
    • 1.进程等待必要性
    • 2.进程等待的方法
    • 3.获取子进程status
    • 4.思考问题
  • 4.进程程序替换
    • 0.为什么要创建一个新的子进程?
    • 1.替换原理
    • 2.替换函数
    • 3.思考问题

1.进程创建

fork_3">1.fork函数初识

  • Linuxfork函数是非常重要的函数,它从已存在进程中创建一个新进程

    • 新进程为子进程,而原进程为父进程
  • 进程调用fork,当控制转移到内核中的fork代码后,内核做:

    • 分配新的内存块和内核数据结构给子进程
      • 必须子进程自己独有,因为进程具有独立性
    • 将父进程部分数据结构内容拷贝至子进程
      • 代码:都是不可被写的,只能读取,所以父子共享,没有问题
      • 数据:可能被修改的,所以必须分离
    • 添加子进程到系统进程列表当中
    • fork返回,开始调度器调度
      请添加图片描述
  • 当一个进程调用fork之后,就有两个二进制代码相同的进程。而且它们都运行到相同的地方。但每个进程都将可以开始它们自己的旅程

int main(void)
{pid_t pid;printf("Before: pid is %d\n", getpid());if ((pid = fork()) == -1)perror("fork()"), exit(1);printf("After:pid is %d, fork return %d\n", getpid(), pid);sleep(1);return 0;
}
运行结果  
Before: pid is 43676
After:pid is 43676, fork return 43677
After:pid is 43677, fork return 0
  • 进程43676先打印before消息,然后它有打印after。另一个after消息由43677打印的
    • 注意到进程43677没有打印before,为什么呢?

请添加图片描述

  • 所以,fork之前父进程独立执行,fork之后,父子两个执行流分别执行
    • 注意fork之后,谁先执行完全由调度器决定

fork_46">2.fork函数返回值

  • 父进程返回的是子进程的PID
  • 子进程返回0

3.写时拷贝

  • 通常,父子代码共享,父子在不写入时,数据也是共享的,当任意一方试图写入,便以写时拷贝的方式各自获得一份副本
  • 为什么要这样设计呢?为什么不可以在创建进程的时候,就直接拷贝分离父子进程呢?
    • 可能拷贝子进程根本就不会用到数据空间,即便是用到了,也可能只是读取
    • 创建子进程,不需要将不会被访问的数据或者只会被读取的数据拷贝一份
    • 但是,还必须得拷贝数据,但什么样的数据值得拷贝呢?
      • 将来会被父/子进程写入的数据
    • 但是,提前拷贝了,会立马使用吗?
      • 一般而言,即便是OS,也无法提前知道哪些空间可能会被写入
    • 综上,OS选择了"写时拷贝"技术,来进程将父子进程的数据分离
      • 因为有写时拷贝的存在,所以,父子进程得以彻底分离,完成了进程独立性的技术保证
  • OS为何要选择写时拷贝技术,对父子进程进行分离?
    • 用的时候,再给你分配,是高效使用内存的一种表现
    • OS无法在代码执行前预知哪些空间会被访问

请添加图片描述

fork_68">4.fork之后,父子进程代码共享

  • 共享的代码是after之后的代码,还是所有的代码都共享呢?
    • 共享的是所有的代码
    • 但是执行的是fork之后的代码

fork_73">5.fork常规用法

  • 一个父进程希望复制自己,使父子进程同时执行不同的代码段
    • 例如:父进程等待客户端请求,生成子进程来处理请求
  • 一个进程要执行一个不同的程序
    • 例如:子进程从fork返回后,调用exec函数

fork_79">6.fork调用失败的原因

  • 系统中有太多的进程
  • 实际用户的进程数超过了限制

2.进程终止

0.进程终止时,操作系统做了什么?

  • 释放进程申请的相关内核数据结构和对应的数据和代码 --> 本质就是释放资源

1.进程退出场景

  • 代码运行完毕,结果正确
  • 代码运行完毕,结果不正确
  • 代码没有跑完,程序崩溃了
  • main函数的返回值的意义是什么?return 0的含义是什么?为什么总是0?
    • 返回给上一级进程,用来评判该进程执行结果用的,可以忽略
    • 此处的0为进程的**退出码,**并不总是0
      • 0:success 非0:标识的时运行的结果不正确
      • 非零值有无数个,不同的非零值就可以标识不同的错误原因
      • 给我们的程序在运行结束之后,结果不正确时,方便定位错误的原因细节
      • 程序崩溃的时候,退出码无意义 – 一般而言退出码对应的return语句没有被执行

2.进程常见退出方法

  • 可以通过echo $?查看进程退出码
  • 正常终止:
    • 从main函数,return 退出码
      • return语句就是终止进程的
      • return n等同于执行exit(n)
    • 调用exit
      • exit在代码的任何地方调用,都表示直接终止进程
    • _exit
  • 异常退出:
    • Ctrl + c,信号终止

4 _exit函数(系统接口)

void _exit(int status);
  • 参数:status定义了进程的终止状态,父进程通过wait来获取该值
    • **说明:虽然status是int,但是仅有低8位可以被父进程所用。所以_exit(-1)时,在终端执行$?**发现返回值是255

4.exit函数(库函数)

void exit(int status);
  • exit最后也会调用**_exit**(库函数封装系统接口)**,**但在调用之前,还做了其他工作:
    1. 执行用户通过atexiton_exit定义的清理函数
    2. 关闭所有打开的流,所有的缓存数据均被写入
    3. 调用_exit
      请添加图片描述
int main()
{printf("hello");exit(0);
}// 运行结果:hello
int main()
{printf("hello");_exit(0);
}// 运行结果:(空)

3.进程等待

1.进程等待必要性

  • 子进程退出,父进程若不管不顾,就可能造成"僵尸进程"的问题,进而造成内存泄漏
  • 另外,进程一旦变成僵尸状态,那就刀枪不入,"杀人不眨眼"的kill -9 也无能为力
  • 因为谁也没有办法杀死一个已经死去的进程
  • 最后,父进程派给子进程的任务完成的如何,我们需要知道
  • 如:子进程运行完成,结果对还是不对, 或者是否正常退出
  • 父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息

2.进程等待的方法

  • wait: pid_t wait(int *status);
    • 返回值:成功返回被等待进程pid,失败返回-1
    • 参数:输出型参数,获取子进程退出状态,不关心则可以设置成为NULL
  • waitpid: pid_t waitpid(pid_t pid, int *status, int options);
    • 返回值:
      • 收集到的子进程的进程ID
        • 等待成功&&子进程退出
      • 如果设置了选项WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0
        • 等待成功&&子进程未退出
      • 如果调用中出错,则返回**-1**
    • 参数:
      • pid:
        • Pid=-1,等待任一个子进程 --> 与wait等效
        • Pid>0,等待其进程ID与pid相等的子进程
      • status:
        • WIFEXITED(status):若为正常终止子进程返回的状态,则为真(查看进程是否是正常退出
        • WEXITSTATUS(status):若WIFEXITED非零,提取子进程退出码(查看进程的退出码
      • options:
        • WNOHANG:让父进程非阻塞等待
          • 若pid指定的子进程没有结束,则waitpid()函数返回0
        • 默认为0,表示阻塞等待
  • 注意:
    • 如果子进程已经退出,调用wait/waitpid时,wait/waitpid会立即返回,并且释放资源,获得子进程退出信息
    • 如果在任意时刻调用wait/waitpid,子进程存在且正常运行,则进程可能阻塞
    • 如果不存在该子进程,则立即出错返回

请添加图片描述

3.获取子进程status

  • wait和waitpid,都有一个status参数,该参数是一个输出型参数,由操作系统填充

    • 如果传递NULL,表示不关心子进程的退出状态信息
    • 否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程
  • status****不能简单的当作整形来看待,可以当作位图来看待(只研究status低16比特位)

    • 次低8位,表示子进程退出的退出码
    • 最低7位,表示进程收到的信号
    • 第8位,core dump标志
      请添加图片描述
  • 由此,可知:

    • 进程异常退出或者崩溃,本质是操作系统杀掉了进程
    • 操作系统如何杀掉进程的呢?
      • 本质是通过发送信号的方式
    • 程序异常,不光光是内部代码有问题,也可能是外力直接杀掉
      • 子进程跑完了吗? --> 不能确定

4.思考问题

  • 父进程为什么要用wait/waitpid函数拿子进程的退出结果?直接全局变量不行吗?
    • 进程具有独立性,数据会进行写时拷贝,父进程无法拿到子进程修改过的数据
    • 并且,信号无法通过全局变量拿到
  • 既然进程是具有独立性的,进程退出码也是子进程的数据,父进程为什么能拿到?wait/waitpid究竟干了什么?
    • 僵尸进程:至少要保留该进程的PCB信息
      • task_struct里面保留了任何进程退出时的退出结果信息
    • 本质就是读取子进程的task_struct结构
    • wait/wait有权利拿task_struct里面的数据么?task_struct是内核数据结构对象
      • wait/waitpid是系统调用,操作系统当然可以访问内核数据

4.进程程序替换

0.为什么要创建一个新的子进程?

  • 为了不影响父进程
  • 想让父进程聚焦在读取数据,解析数据,指派进程执行代码的功能
  • 如果不创建,那么替换的进程只能是父进程,如果创建了,替换的进程就是子进程,而不影响父进程

1.替换原理

  • fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支)
    • 但如果子进程想执行一个全新的程序,该怎么办?
  • 子进程往往要调用一种exec函数以执行另一个程序
    • 当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行
      • 将新的磁盘上的程序加载到内存,并和当前进程的页表,重新建立映射
    • 调用exec并不创建新进程,所以调用exec前后该进程的id并未改变

请添加图片描述

2.替换函数

  • 有六种以exec开头的函数,统称exec函数
    • exec*:功能其实就是加载器的底层接口
int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ..., char *const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execve(const char *path, char *const argv[], char *const envp[]);
  • 可变参数列表部分/环境变量,最后一个参数必须是NULL,标识参数传递完毕
  • 这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回
    • 如果调用出错则返回-1
    • 所以exec函数只有出错的返回值而没有成功的返回值
  • 命名理解:
    • l(list) : 表示参数采用列表
    • v(vector) : 参数用数组
    • p(path) : 有p自动搜索环境变量PATH
      • 在环境变量PATH中查找要执行的程序
    • e(env) : 表示自己维护环境变量
  • 注意:传参时,第一个参数都应该和要替换的程序名一样
函数名参数格式是否带路径是否使用当前环境变量
execl列表不是
execlp列表
execle列表不是不是,须自己配置环境变量
execv数组不是
execvp数组
execve数组不是不是,须自己配置环境变量
  • 事实上,只有execve是真正的系统调用,其它五个函数最终都调用execve

请添加图片描述

3.思考问题

  • 加载新程序之前,父子的数据和和代码的关系 --> 代码共享,数据写时拷贝
    • 当子进程加载新程序的时候,不就是一种"写入"么,代码要不要写时拷贝,将父子的代码分离?
      • 必须分离
      • 父子进程在代码和数据上就彻底分开了

http://www.ppmy.cn/ops/1838.html

相关文章

JVM修炼之路【11】- 解决内存溢出、内存泄漏 以及相关案例

前面的10篇 都是基础的知识,包括类加载的过程 类加载的细节,jvm内存模型 垃圾回收 等等, 这一篇我们开始实战了解一下 各种疑难杂症:怎么监控 怎么发现 怎么解决 内存溢出 内存泄漏 这两个概念在垃圾回收器里面已经讲过了&#…

第10章 物理安全要求

10.1 站点与设施设计的安全原则 假如没有对物理环境的控制,任何管理的、技术的或逻辑的访问控制技术都无法提供足够的安全性。 如果怀有恶意的人员获取了对设施及设备的物理访问权,那么他们几乎可以为所欲为,包括肆意破坏或窃取、更改数据。…

Spark-Scala语言实战(15)

在之前的文章中,我们学习了如何在spark中使用键值对中的学习键值对方法中的lookup,cogroup两种方法。想了解的朋友可以查看这篇文章。同时,希望我的文章能帮助到你,如果觉得我的文章写的不错,请留下你宝贵的点赞&#…

FreGS:具有渐进频率正则化的3D高斯溅射

FreGS: 3D Gaussian Splatting with Progressive Frequency Regularization FreGS:具有渐进频率正则化的3D高斯溅射 Jiahui Zhang1  Fangneng Zhan2  Muyu Xu1  Shijian Lu1  Eric Xing3, 4 张家慧 1 詹方能 2 许慕玉 1 卢世坚 1 邢志伟 3, 4 1Nanyang Technolo…

时隔一年,再次讨论下AutoGPT-安装篇

AutoGPT是23年3月份推出的,距今已经1年多的时间了。刚推出时,我们还只能通过命令行使用AutoGPT的能力,但现在,我们不仅可以基于AutoGPT创建自己的Agent,我们还可以通过Web页面与我们创建的Agent进行聊天。这次的AutoGP…

策略模式(Strategy Pattern)在JAVA中的应用

设计模式是软件工程中的一套被广泛认可的解决特定问题的最佳实践。它们是在多年的软件开发实践中总结出的有效方法。策略模式是JAVA中常用的一种行为型设计模式,它定义了一系列算法,并将每一个算法封装起来,使它们可以互换,让算法…

Elasticsearch:(二)3.集群的健康检查

1.健康状态 green:所有primary主分片和replica副分片均为active,集群健康。 yellow:至少一个replica副本分片不可用,但是所有primary主分片均为active,数据仍然是可以保证完整性的。 red:至少有一个primary为不可用状态,数据不…

uniapp 开发小程序如何检测到更新点击重启小程序完成更新?

官方文档:uni.getUpdateManager() | uni-app官网 示例代码: const updateManager uni.getUpdateManager();updateManager.onCheckForUpdate(function (res) {// 请求完新版本信息的回调console.log(res.hasUpdate); });updateManager.onUpdateReady(fu…

HTML5新增的多媒体标签

在网页中加入音乐 <audio></audio> src 设置音乐文件名以及路径,<audio>标记支持MP3、WAV及OGG 3种音乐格式 autoplay&#xff1a;是否自动播放,加入autopaly属性表示自动播放 controls&#xff1a; 是否显示播放面板,加入controls属性表示显示播放面板 …

比特币叙事大转向

作者&#xff1a;David Lawant 编译&#xff1a;秦晋 要理比特币解减半动态&#xff0c;最关键的图表是下面这张&#xff0c;而不是价格图表。它显示了自 2012 年以来&#xff0c;矿业总收入与比特币现货交易量的比例&#xff0c;并标注了三个减半日期。 虽然矿工仍然是比特币生…

租用境外服务器,越南服务器的优势有哪些

自从中国加入世界贸易组织之后&#xff0c;国内经济增加速度非常快&#xff0c;同时越来越多的人选择去东南亚国家发展&#xff0c;因为当地的中国人很多&#xff0c;所以中国企业在当地面临着更小的文化差异。东南亚地区也是最新的经济体&#xff0c;互联网正处于蓬勃发展的阶…

Matlab方程组拟合【案例源码+视频教程】

专栏导读 作者简介&#xff1a;工学博士&#xff0c;高级工程师&#xff0c;专注于工业软件算法研究本文已收录于专栏&#xff1a;《复杂函数拟合案例分享》本专栏旨在提供 1.以案例的形式讲解各类复杂函数拟合的程序实现方法&#xff0c;并提供所有案例完整源码&#xff1b;2.…

ElasticSearch(ES)语法关键词

文章目录 第一级关键词第二级关键词query使用highlight使用settings使用 第三级关键词boolanalysis使用 其他参考文档 第一级关键词 query&#xff1a;查询条件from&#xff1a;分页偏移量size&#xff1a;分页数量sort&#xff1a;排序_source&#xff1a;选择需要的字段aggs…

pytorch 今日小知识1——torch.set_printoptions

torch.set_printoptions torch.set_printoptions(precisionNone, thresholdNone, edgeitemsNone, linewidthNone, profileNone, sci_modeNone)这个方法是来修改pytorch中的打印选项的&#xff08;其实就是和numpy中更改打印方法一样&#xff09;&#xff0c;就是使用print打印…

“好”玩游戏让我本能痴迷游戏编程

源地址&#xff1a;https://www.ctvol.com/c-cdevelopment/5842.html 我的游戏生涯是从最开始的热血传奇开始的&#xff0c;那时候&#xff0c;我们年少轻狂&#xff0c;不知道多少80后的青春都洒在了这个游戏上面&#xff0c;那时候&#xff0c;热血传奇的热度比现在的英雄联…

kafka快速入门+应用

Kafka, 构建TB级异步消息系统 0.mq优劣 优点&#xff1a;应用解耦、异步提速、削峰填谷 缺点&#xff1a;系统可用性降低、系统复杂度提高、一致性问题 mq特点&#xff1a;消息不丢失、支持消息存储 1.快速入门 1.1 阻塞队列 在生产线程 和 消费线程 之间起到了 &#xff…

SpringSecurity集成JWT

使用 Spring Security 集成 JWT&#xff08;JSON Web Token&#xff09;身份验证是一种常见的方式来实现基于令牌的身份验证。在 Spring Boot 应用程序中使用 Spring Security 和 JWT&#xff0c;可以创建一个安全、可扩展的身份验证系统。下面是一个示例&#xff0c;展示如何在…

虚拟现实(VR)开发框架

虚拟现实&#xff08;VR&#xff09;开发框架为开发者提供了构建VR应用程序所需的基本工具和功能。它们通常包括3D引擎、场景图、输入系统、音频系统和网络功能。下面是一些流行的VR开发框架。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流…

C语言 | Leetcode C语言题解之第24题两两交换链表中的节点

题目&#xff1a; 题解&#xff1a; struct ListNode* swapPairs(struct ListNode* head) {struct ListNode dummyHead;dummyHead.next head;struct ListNode* temp &dummyHead;while (temp->next ! NULL && temp->next->next ! NULL) {struct ListNod…

变更docker的缓存路径

承接上一篇文章&#xff0c;定期清理docker&#xff0c;仍感觉有些累&#xff0c;这个变更一下缓存路径&#xff0c;可以根本解决问题。 linux磁盘清理_docker/overlay2爆满_linux overlay目录满了-CSDN博客 当然&#xff0c;变更docker的缓存路径&#xff0c;也可以使用上一…