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

linux-----进程控制

提示:以下是本篇文章正文内容,下面案例可供参考

一、fork()函数

返回值:子进程返回0,父进程返回子进程的id,出错就返回-1.

 fork创建子进程,如果父子一方发生写入时,就会发生写实拷贝,操作系统就会重新申请空间,发生拷贝,修改页表,构建新的虚拟到物理的映射关系。

发生写实拷贝的过程中是怎么样的,fork创建子进程时,操作系统会把父进程的页表项都改为只读,子进程拷贝了一份父进程的页表,发生写入时,由于页表项都是只读的权限,就会因为权限问题出错,这时操作系统就会介入,查看是什么原因出错,例如栈堆区的数据发生修改等,由于这个地方的数据原来就是可读可写的,那么就会发生写实拷贝,重新申请空间,构建虚拟到物理的映射,如果是由于只读常量区发生修改,这个地方本来就是可读不可写的,那么程序就会出错。

我们可以创建多个进程来帮我们执行任务:记得让父进程等待子进程,子进程退出了,它的数据和内存就释放了,不过管理这个进程的PCB还没有释放,就会有大量的僵尸进程,消耗系统内存资源,变卡。

二、进程的中止

1.进程的退出方法

除了main函数return可以退出,还有exit和_exit.它两之间还是有区别的 。exit库函数封装了_exit.exit是库函数而_exit是系统调用。mian函数返回时也会调用了exit.所有也会刷新缓冲区的数据

exit会刷新缓冲区,而_exit不会刷新。

这里我们可以预想一下,其实输出输出都有缓冲区,一般输出时带个\n就会刷新缓冲区,还有就是main函数return返回时也会刷新缓冲区,可是_exit是系统系统调用,exit是函数调用,说明这个缓冲区不是内核缓冲区。 

验证:先不让父进程退出。第一个子进程应该是休眠十秒后,刷新缓冲区,显示打印的结果,而不是打印时直接显示到屏幕上。第二个由于没有休眠return直接刷新缓冲区。结果应该是“hello _exit”休眠十秒后显示第一个子进程要打印的内容。

2.进程退出码和错误码

错误码:这个是在c语言中,是个全局变量。如果库函数调用出错,那么就会把错误码设置,我们要知道错误的原因可以使用strerror打印。这里我们打印了0-9错误码的代表的含义

进程退出码:是进程退出也就是main函数的返回值(注意区分函数中的return),在bash中我们可以拿到最近一次程序的退出码,通过命令获取:echo $?。(echo命令也是个程序,bash通过创建子进程来执行的)。

一个程序退出一共有三种状态,代码跑完,结果正确,代码跑完,结果不正确,程序异常中止,代码没跑完。通过退出码获取进程执行的结果

退出码为0,代表程序没出错,跑完了。一般退出码非0,代表程序出问题了,非0的情况一般分为两种,我们自己定义的退出码,还有一种是操作系统对进程发出信号中止进程的退出码,比如除零错误。我们还可以根据自己的需求自己定义退出码,根据退出码打印原因。    

这个由于除零异常,操作系统直接发信号给进程,提前退出了。bash是父进程,

通过创建子进程来执行./a.out.bash拿到子进程的退出码,然后解释退出的原因。 

进程异常退出是收到了信号,我们可以给进程发信号,即使这个程序是正常运行的。我们给进程发个浮点数异常的信号(8号信号SIGFPE)。

总结:一个程序出异常肯定是收到了操作系统给进程发出的信号,如果没出异常代码跑完了,那么我们可以通过进程的退出码,来判断程序的执行的结果的状态,结果是否符合我们的预想。

3.进程等待

(1)wait()

 等待父进程的任意一个子进程的退出。就是父进程哪个子进程先退出,就等待哪个。

(2)waitpid()

第二个函数的第一个参数是子进程的pid,指定等待哪个子进程退出。int* wstatus是个输出型参数,通过它可以拿到进程的退出码,第三个参数是设置是否堵塞式等待。注意的是,如果options为零,它是个堵塞式等待函数,只有等待成功(条件:子进程退出状态为僵尸)函数返回值为被等待的子进程的pid,才会向下执行,如果options不为零,就是非堵塞式等待了。

大多情况下下,我们使用这个非堵塞式等待,如果刚好子进程退出了,那么它的返回值为子进程的pid,如果子进程没退出(条件没具备:子进程没退出,状态不是僵尸)函数返回值为0,如果返回值小于0说明等待失败了。要注意的是,不管有没有等待成功(返回值为0或者为子进程的pid),它都会向下执行父进程的代码,不会堵塞式等待成功后,才去执行后面的代码。

这里我们用非堵塞式等待

如何等待一批进程然后进行回收。一种是记录子进程的pid,然后根据子进程的数量,一个个等待。

不过我们一般不知道哪个进程先退出,效率不高。直接将waitpid的第一个参数设为-1. 表示等待任意一个进程退出,这个和wait比较像。

 ​​​​​​​

4.如何获取进程的status

status是个int类型的参数,通过函数调用拿到进程PCB中status,不过我们主要看后面的低16位,它可以记录退出码和退出信号。进程退出时或者中止时,操作系统会通过这个进程PCB中status,记录退出码或者收到的信号,同时将进程状态修改为僵尸。

如果进程正常终止后八位无意义,进程异常中止前8位无意义。

我们可以通过等待函数的输出型参数拿到这个数字,获取进程退出的状态。

当然我们直接通过宏来直接获取子进程的退出码

 


http://www.mrgr.cn/news/33808.html

相关文章:

  • 妈妈再也不用担心字符串方法啦!——js String实例方法汇总
  • 分布式安装LNMP
  • 基于 Web 的工业设备监测系统:非功能性需求与标准化数据访问机制的架构设计
  • 传输层 III(TCP协议——可靠传输)【★★★★】
  • 【Spring 底层原理】手搓一个Spring框架
  • 【busybox记录】【shell指令】numfmt
  • 嵌入式系统基础讲解
  • 用apache httpd来实现反向代理
  • golang学习笔记3-变量的声明
  • CORS跨域+Nginx配置、Apache配置
  • 2024.9.22
  • screen使用——关机时在服务器上跑代码
  • 蓝桥杯嵌入式的学习总结
  • UE学习篇ContentExample解读-----------Blueprint_Overview
  • 《深度学习》—— 卷积神经网络(CNN)的简单介绍和工作原理
  • 深度学习与应用:人体关键点检测
  • SpringBoot项目License证书生成与验证(TrueLicense) 【记录】
  • 一种求解无人机三维路径规划的高维多目标优化算法,MATLAB代码
  • Java-Part 0
  • 2009考研数学真题解析-数二: