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

进程控制(上)【Linux操作系统】

进程控制

写时拷贝

本质是一种减少深拷贝的方法
Linux中有很多拷贝的场景都用得上写时拷贝,下面以创建子进程时的写时拷贝为例:

子进程被创建的时候:
会继承父进程的mm_struct和页表
所以子进程刚刚继承时,父子进程的代码和数据都是共享的

系统是如何知道子进程/父进程修改数据的时候要发生写时拷贝的呢?
①父进程在使用fork创建子进程之前,就会把页表中的数据区的权限改成r(只读)
这样子进程继承到的页表中的数据区就也是只读的

②当子进程/父进程尝试修改数据区中的数据时,就是修改只读数据项,页表就会报错,就会触发缺页中断

③系统发现缺页中断之后,就会检测
1,如果发现用户要修改的这个区域一定是只读的,就把进程杀掉
2,如果这个区域一定是读写的,只是页表中的权限设置成只读的了,此时系统就会进行写时拷贝
在这里插入图片描述

所以:不止创建子进程写时拷贝的时候是这种原理
只要可能发生写时拷贝的数据被页表管理着,就都可以通过这个原理实现写时拷贝



错误码

用来判断进程任务执行是否成功,如果失败了错误是什么

main函数的返回值其实是错误码,是返回给父进程/操作系统的
在这里插入图片描述
进程错误码的的取值范围是[0-255],即一个字节
因为使用wait/waitpid等待回收错误码的时候,只给它留了8个比特位



进程中止

进程中止的方法:

①在main函数中return

②在任何地方使用exit函数[exit函数的参数就是错误码,它的头文件是stdio.h]

③_exit使用方法和exit一模一样


exit与_exit的区别:

本质区别就是exit是用户层的库函数,_exit是内核层的系统调用
所以exit函数中,调用_exit中止进程之前,可以进行一些用户层的收尾工作
比如:
exit中止进程时,会刷新用户级缓冲区和内核级缓冲区
也就是如果输出缓冲区中有数据,使用exit中止它会帮我们打印出来

_exit中止进程时,不会刷新用户级缓冲区,只会刷新内核级缓冲区

②exit是库里面的函数,是操作系统之外的操作,使用操作系统的资源时,只能对操作系统调用接口进行封装
exit其实就封装了_exit

_exit是系统调用接口,是操作系统内部的操作
在这里插入图片描述

所以:
其实我们之前再使用c/c++等高级语言时,所说的缓冲区都是语言级(用户级)的缓冲区

这个缓冲区里面的内容是存储在c/c++的标准库中的,即存储在共享区的

不是存储在操作系统中的内核级缓冲区
不然_exit中止进程时,就也能把缓冲区中信息输出/输入
但是因为语言级缓冲区在操作系统的上层,所以_exit中止进程后,内核缓冲区就关闭了,语言缓冲区中的数据没办法进入操作系统了
操作系统没看见语言缓冲区中传来信息,就以为里面没有



进程等待

wait(不常用)

头文件:sys/types.hsys/wait.h
返回值:pid_t[小于0说明回收失败,大于0时回收成功,返回的是回收的子进程的pid]
参数:

作用:等待并回收任意一个子进程
在这里插入图片描述


waitpid[常用]

头文件:sys/types.hsys/wait.h
返回值:pid_t n
①n小于0,说明回收失败/函数调用失败
②n大于0,回收成功,返回的是成功回收的子进程的pid
③n等于0(只有非阻塞等待才会出现),则表示函数调用成功,但是子进程还没执行完,还没退出

参数1.pid_t pid

如果是子进程的pid,就表示指定等待这个子进程
如果是-1,就表示等待任意一个进程


参数2,int* status

如果传nullptr则表示不需要获取退出信息==

即:用户自己定义一个int类型的变量
再把它的地址传进waitpid里面,操作系统就会把进程的PCB中存储的退出信息给status

status其实并不是一个整数,而是一个类似位图的东西
因为进程结束,分两种情况:
①正常退出,也就是通过main函数的return或者exit退出,这样就可以返回退出码
所以只有正常退出才能返回退出码(错误码)

②异常退出,也就是进程运行的途中,空指针访问/野指针等直接导致进程崩溃了
这样进程根本就运行不到返回退出码的地方
但是异常退出,也会有自己的退出信号码
[通过kill -l命令,可以看到Linux中所有的退出信号以及其对应的退出信号码]

进程之所以会异常退出,是因为进程运行时出现了较严重的错误(野指针,除0等
操作系统识别到以后,在代码还没跑完的时候,就直接使用信号中止了进程


所以
status中不仅仅存储了退出码,还存储了其他的退出信息

具体的:
status的:
①最低的7个比特位存储退出信号值
②第8个比特位存储core dump标志
③第9~16个比特位存储退出码
在这里插入图片描述

宏:
WIFEXITED(status):
若为正常终止子进程返回的状态,则为真(查看进程是否是正常退出)

WEXITSTATUS(status):
若WIFEXITED非零,提取子进程退出码。(查看进程的退出码)


参数3.标识是阻塞等待,还是非阻塞等待

如果该参数是0,表示阻塞式等待
如果为WNOHANG,表示非阻塞式等待



使用子进程完成任务的好处

①非阻塞等待或者创建多个子进程时,父子进程可以同时运行,父进程不用等子进程,提高并发度

②进程具有独立性,所以如果子进程出了问题,不影响父进程

③数据快照,子进程被fork出来,继承了父进程的PCB,页表等东西之后
就获取到了fork时父进程的数据
也就是对子进程从父进程那里获取到的数据进行了快照
快照之后,其他任何进程对这一块数据进行修改,这个子进程都“看不见”
因为写时拷贝,或者说进程具有独立性。所以数据快照之后,其他进程可以对这个数据任意修改

相关文章:

  • 【leetcode100】整数拆分
  • 带宽”(bandwidth)有以下两种不同的意义
  • 41.[前端开发-JavaScript高级]Day06-原型关系图-ES6类的使用-ES6转ES5
  • 【C++游戏引擎开发】第16篇:ImGui指南
  • MTK-Android12 13 屏蔽掉Viewing full screen
  • 【数据结构_9】栈和队列
  • spring:注解@PostConstruct、@PreDestroy
  • 同步 vs 异步 最直白的区别
  • Github webhook 是什么以及它的用途
  • 强化学习的数学原理(四)Value iteration Policy iteration
  • LLamaIndex中经常使用的四个模块
  • 2024年第九届团队程序设计天梯赛c++题解L1-L3-1(附PTA网址)
  • 6 CMD 与 PowerShell 指令大全、C 程序终端运行、字符编码切换指南
  • 【AI工具】用大模型生成脑图初试
  • C++学习记录:
  • 2025主流报表软件评测和推荐
  • 【DeepSeek答】如何成为一名科技领域陪同口译,阶段性学习目标是什么
  • FreeRTOS之消息队列
  • Spring Boot 应用程序中配置使用consul
  • 20250416在荣品的PRO-RK3566开发板的Android13下编译native C的应用程序的步骤
  • 基于b s结构做的网站/许昌网络推广外包
  • 深圳做网站/如何制作简单的网页链接
  • 找人设计的网站/新冠咳嗽一般要咳多少天
  • 提供网站制作公司电话/seo优化的方法有哪些
  • 公司网站有哪些/软件开发外包公司
  • 做网站模板用什么框架/手机自动排名次的软件