[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/embedded/1806.html

相关文章

vim卡死了,没有反应怎么办?

解决办法: 很有可能是你有个在window下的好习惯,没事儿就ctrl s保存文件。但是在vim里,ctrl s默认是发送一种流控制信号,通常用于停止终端的输出,所以你的屏幕就卡死了。 解决办法也很简单,按下ctrl q即…

从IoTDB的发展回顾时序数据库演进史

面向工业物联网时代,以 IoTDB 为代表的时序数据库加速发展。 时序数据的主要产生来源之一是设备与传感器,具有监测点多、采样频率高、存储数据量大等多类不同于其他数据类型的特性,从而导致数据库在实现高通量写入、存储成本、实时查询等多个…

【C++】C++11右值引用

👀樊梓慕:个人主页 🎥个人专栏:《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》《实训项目》《C》《Linux》《算法》 🌝每一个不曾起舞的日子,都是对生命的辜负 目录 前言 1.什么是左值&&…

TCP/IP_第八章_静态路由_实验案例二

实验案例二:配置静态路由实现路由选路 1、实验环境 如图8.11所示,三台路由器R1, R2,R3两两互连。R2上配置了Loopback地址192.168.20 .1/24,模拟192.168.20.0/24网段;R3上配置了两个Loopback地址192.168.10.1/24、192.…

服务器安装完SqlServer远程电脑连接不了

1、将服务器的TCP/IP启用 2、重新启动服务 cmd输入services.msc

测试JAVA 测开

测试、java测开 1、测试用例要素(4个重要要素)2、测试用例的好处3、测试用例的设计方法3.1 基于需求设计测试用例3.2 等价类3.3 边界值3.4 判定表 1、测试用例要素(4个重要要素) 测试环境操作步骤测试数据预期结果 2、测试用例的…

探究C++20协程(2)——取值、传值、销毁与序列生成器实现

序列生成器是一个非常经典的协程应用场景,尤其是在需要惰性生成数据或处理潜在无限的数据流时。 序列生成器概念:序列生成器允许程序按需生成序列中的下一个元素,而不是一次性计算整个序列。这种方式可以节省内存,并允许处理无限或未知长度的…

Django之rest_framework(三)

一、GenericAPIView的使用 rest_framework.generics.GenericAPIView 继承自APIVIew,主要增加了操作序列化器和数据库查询的方法,作用是为下面Mixin扩展类的执行提供方法支持。通常在使用时,可搭配一个或多个Mixin扩展类 1.1、属性 serializer_class 指明视图使用的序列化器…

JavaWeb--JavaScript-事件绑定/BOM/DOM编程

目录 1. 事件绑定 1.1. 什么是事件 1.2. 常见事件 1.3. 事件的绑定 1.3.1. 属性绑定 1.3.2. DOM编程绑定 1.4. 事件的触发 1.4.1. 行为触发 1.4.2. DOM编程触发 2. BOM 编程 2.1. 什么是 BOM 2.2. window对象的常见属性(了解) 2.3. window对象的常见方法(了解) 2…

Qt_30道常见面试题及答案

1. 简述 Qt 是什么? 答:Qt 是一个跨平台的应用程序开发框架,它提供了一系列的工具和库,用于开发图形用户界面(GUI)应用程序。 2. Qt 有哪些主要模块? 答:Qt 的主要模块包括 Qt Co…

读《AI营销画布》步骤五 保收获(十)

前言 正如书中所说,做到前四步就已经很了不起了,但是,现如今有很多公司的IT部门正从原来的公司分离,成立了不同的科技公司,以确保从费用成本中心变为利润中心,当然,分离出来不一定是AI促进的&am…

【经典算法】LeetCode 64. 最小路径和(Java/C/Python3/Golang实现含注释说明,Easy)

作者主页: 🔗进朱者赤的博客 精选专栏:🔗经典算法 作者简介:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名) ❤️觉得文章还…

【ElasticSearch】安装(bug篇)

以下解决办法参考自网友们的分享 1. JDK绑定问题 但其实这样也没有问题,因为内嵌的jdk版本与当前的es版本是适配的 但是,如果内嵌的jdk与当前es不适配,那就要修改配置文件 / 添加环境变量,让es启动的时候能扫描到我们本地的jdk …

已解决java.net.NoRouteToHostException: 无法到达主机异常的正确解决方法,亲测有效!!!

已解决java.net.NoRouteToHostException: 无法到达主机异常的正确解决方法,亲测有效!!! 目录 问题分析 报错原因 解决思路 解决方法 检查网络连接 核实目标地址 检查防火墙和路由器规则 验证VPN/代理设置 修正网络配置 …

算法库应用-有序单链表插入节点

学习源头: 模仿贺利坚老师单链表排序文章浏览阅读5.9k次。  本文针对数据结构基础系列网络课程(2):线性表中第11课时单链表应用举例。例:拆分单链表 (linklist.h是单链表“算法库”中的头文件,详情单击链接…)//本程…

工业级POE交换机的ACL

工业级POE交换机通常支持访问控制列表(Access Control List,ACL)功能,用于实施网络安全策略。ACL可以根据源IP地址、目标IP地址、传输协议、端口号等条件来过滤和控制网络流量。 通过配置ACL,可以实现以下功能&#xf…

docker 容器迁移

目录 1、将容器打成镜像后迁移 2、导出和导入容器 1、将容器打成镜像后迁移 (1)将容器打成镜像 # 打成镜像 mycentos docker commit -m "my centos" -a "author" 2d1fba0978 mycentos # 打成镜像 mycentos,tag …

如何花少量的钱举办一场漂亮的知识竞赛

如果预算不多,是不是就不能办出一场漂亮的知识竞赛活动了呢。我看未必,大家可以从以下几个方面去考虑。 一、降低场地费用 知识竞赛活动的费用大头是场地及配套设备(大屏、音响、灯光、舞台等),先要考虑如何省去或减…

MySQL 锁机制全面解析

目录 1. MySQL的锁类型1.1 全局锁1.2 表锁1.3 行锁1.4 共享锁(读锁)1.5 排它锁(写锁)1.6 死锁 2 乐观锁和悲观锁2.1 乐观锁2.2 悲观锁 3 意向锁4 间隙锁5 临键锁6. 事务隔离级别对锁的影响6.1 读未提交(Read Uncommitt…

ASP.NET基于BS方式的即时通讯软件的设计与实现

摘 要 即时通讯(Instant Messaging)是目前Internet上最为流行的通讯方式,而各种各样的即时通讯软件也层出不穷;服务提供商也提供了越来越丰富的通讯服务功能。随着互联网的发展,即时通讯的运用将日益广泛&#xff0c…