【进程终止】退出信号 | 三种退出情况 | 如何进程终止returnexit_exit

目录

退出码

退出信号

进程终止情况3

如何进程终止

return退出

库函数exit

系统调用函数_exit

​exit和_exit的区别缓冲区

exit

_exit


退出码

回顾上篇

  • 代码跑完,结果正确(退出码为0)
  • 代码跑完,结果不正确(退出码为非0)
  • 结果不正确退出码有相应的错误描述:系统定义和自定义。
  • 所以进程终止代码跑完,结果正不正确是根据子进程的退出码来决定的。

当然,除此之外还存在第三种情况:代码没跑完。代码执行时,出现了异常,提前退出了。

  1. 先确认是否异常
  2. 不是异常,就一定是代码跑完了,结果是否正确,看退出码

退出信号

子进程执行过程中出现了错误,OS此刻直接把正在执行的进程干掉,不让进程继续执行了。当我们的程序运行(进程)语言层面上是崩溃,系统层面上是OS杀死进程。

  • 语言层面上,编程运行奔溃了
  • 系统层面上,OS杀死了进程,此时退出码就没有意义了。
  • 进程出现异常,我们还是需要知道为什么的?用户也需要知道为什么?
  • 进程出现异常,本质上是因为进程收到了OS发给进程的信号❗(这个过程类似我们用kill -9 进程pid 杀死某个进程的过程)
  • kill -9 进程pid 说明进程如果永远都不退出,但是系统层面上用信号的方式让进程提前终止或者结束循环
  • 查看所有信号:kill -l

  • 野指针和被除数为0等错误操作都会造成进程出现异常
  • int *p = NULL *p =100;(访问0号地址,0号地址进程不能/也没有权力访问,造成了野指针)
  • Segmentation fault段错误,OS提前终止进程。
  • OS:都是野指针了,终止进程,别执行了。OS测进程异常,给它发了信号,终止进程。异常触发了OS给进程发信号。
  • OS系统中常见的报错(奔溃)最终在OS层面上都是OS给指定的进程发送信号,进而终止进程,把错误信息展现给用户。

触发进程信号:

  • 可以手动触发kill -9 进程pid
  • 可以自动触发系统给进程发送信号

综上所述:我们可以通过查看进程终止的时候,退出信号是多少,就可以判断进程异常的原因了❗

int main()37 {38     int *p = NULL;39     int result = Div(10, 100);40     printf("result: %d [%s]\n", result, CodeToErrString(exit_code));41     result = Div(10, 0);42     printf("result: %d [%s]\n", result, CodeToErrString(exit_code));43     *p = 100;//野指针44     return exit_code;                                                                           45 }

进程终止情况3

进程终止存在3种情况:

  • 代码运行完毕,结果正确
  • 代码运行完毕,结果不正确
  • 代码异常终止

衡量一个进程退出,我们只需要两个数字:退出码和退出信号。全部都是由这两个数字组合。这两个数字一定会让父进程知道子进程执行的情况。


联系下篇进程等待

  • 子进程终止了,子进程处于僵尸状态Z状态。
  • 子进程终止后,代码和数据都会被OS释放掉,不会立刻把进程PCB释放掉。
  • 子进程的PCB需要维护一段时间,里面存有子进程退出的数据(退出码和退出信号),等待父进程来读取。 
  • 一个进程退出时,进程的信号/提出码都会写入进程的PCB。
  • 父进程等待就会拿到进程的退出信息,并让用户看到。
  • 退出信息:
  • int sig_code;
  • int exit_code; 

如何进程终止

  • 以下进程终止不考虑进程异常的情况。
  • 进程终止也就是main函数结束了

进程常见的退出方法:

  • 正常终止(可以通过 echo $? 查看进程退出码):
  1. 从main返回
  2. 调用exit
  3. _exit
  • 异常退出:
  1. ctrl + c,信号终止

综下所述:

  • main函数return,表示进程终止(非main函数,return,函数结束)
  • 代码调用exit函数,在代码任意位置调用exit,都表示进程退出。exit的参数就是我们的退出码。不传参数就是系统自带的退出码和退出描述。
  • 系统调用接口函数_exit

return退出

  1. 在main函数中,return表示进程终止
  2. 在非main函数中,return表示函数结束
  • 函数结束并不一定代表进程终止
  • return是一种更常见的退出进程方法。执行return n等同于执行exit(n),因为调用main的运行时函数会将main的返回值当做 exit的参数。
  • main函数return,表示进程终止(非main函数,return,函数结束)
  1 #include<stdio.h>                                                                                        2 int Add(int x,int y)3 {4   return x+y;5 }6 int main()7 {8   int ret=Add(10,20);9   printf("%d\n",ret);10   return 0;11 }

库函数exit

  • 查看exit:man 3 exit(3号手册)
  • exit引起一个正常的进程终止。
  • int status类似退出码 
  • exit在main函数内部可以直接终止进程
  • exit在函数内部也可以直接终止进程
  • #include <unistd.h>
  • void exit(int status);
  • 代码调用exit函数,在代码任意位置调用exit,都表示进程退出。exit的参数就是我们的退出码。不传参数就是系统自带的退出码和退出描述。

exit最后也会调用exit, 但在调用exit之前,还做了其他工作:

  1. 执行用户通过 atexit或on_exit定义的清理函数。
  2. 关闭所有打开的流,所有的缓存数据均被写入
  3. 调用_exit

【exit在main函数】

  1 #include<stdio.h>2 #include<stdlib.h>3 #include<unistd.h>4 int Add(int x,int y)5 {6   return x+y;7 }8 int main()9 {10   int ret=Add(10,20);11   sleep(2);12   exit(7);                                                                                               13   printf("%d\n",ret);14   return 0;15 }

 【exit在函数】

  1 #include<stdio.h>2 #include<stdlib.h>3 #include<unistd.h>4 int Add(int x,int y)5 {6   exit(123);7   return x+y;                                                                                            8 }9 int main()10 {11   int ret=Add(10,20);12   printf("%d\n",ret);13   return 0;14 }

系统调用函数_exit

  • 查看系统调用函数_exit():man 2 _exit(2号手册)
  • 终止一个调用进程
  • 同样_exit()和exit()一样无论在代码任意位置使用都可以终止进程。
  • #include <unistd.h>
  • void _exit(int status);
  • 参数:status 定义了进程的终止状态,父进程通过wait来获取该值
  • 说明:虽然status是int,但是仅有低8位可以被父进程所用。所以_exit(-1)时,在终端执行$?发现返回值是255。
  • ❗_exit和exit的区别是缓冲区

 

exit和_exit的区别缓冲区

  • 综下所述:exit会在进程退出的时候,冲刷缓冲区。_exit不会,把数据丢弃直接终止进程。说明目前我们所说的缓冲区不是内核缓冲区。缓冲区一定不在OS内部维护。 
  • exit是标准C语言封装的库函数
  • _exit是OS给上层用户提供的系统调用接口函数
  • 缓冲区一定不在系统调用接口下面,不在_exit内部/OS内部(因为在的_exit会做刷新)
  • 其实,exit底层就是调用的_exit系统调用接口(终止进程的本质是让进程释放进程的代码/数据所占用的内存资源,释放除进程PCB外的其他内核数据结构,就是对进程做管理的一种方式。用户是没有权力对OS内部的任何字段做任何访问的)
  • 缓冲区一定在_exit之上。

exit

无\n

  • 先休眠再冲刷缓冲区,让缓冲区的数据打印在显示器上。
  • 进程开始执行的时候,打印的结果并未显示出来,sleep(2)时,打印的结果数据放在缓冲区中。
  • 最后exit(3)时,会协助进程,在进程退出的时候冲刷缓冲区的数据到显示器上。
  • 注意exit底层是先冲刷缓冲区再调用系统调用函数_exit函数终止进程。
//先冲刷缓冲区再休眠/n1 #include<stdio.h>2 #include<stdlib.h>3 #include<unistd.h>4 int main()5 {6   printf("Hello linux!\n");7   sleep(2);8   exit(7);                                                                                               9 }
//先休眠再冲刷缓冲区                                                                     1 #include<stdio.h>2 #include<stdlib.h>3 #include<unistd.h>4 int main()5 {6   printf("Hello linux!");7   sleep(2);8   exit(7);                                                                                               9 }

_exit

 无\n时,发现_exit没有刷新缓冲区的数据,就直接终止程序了。

//先冲刷缓冲区再休眠/n1 #include<stdio.h>2 #include<stdlib.h>3 #include<unistd.h>4 int main()5 {6   printf("Hello linux!\n");7   sleep(2);8   _exit(7);                                                                                               9 }
//先休眠再冲刷缓冲区                                                                     1 #include<stdio.h>2 #include<stdlib.h>3 #include<unistd.h>4 int main()5 {6   printf("Hello linux!");7   sleep(2);8   _exit(7);                                                                                               9 }

🙂感谢大家的阅读,若有错误和不足,欢迎指正。因为LinuxOS系统概念很多所以呢要学习更加细致才能写博客会更新的比较慢一些🙂,下篇进程等待。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/833164.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringDI方式及Redis应用场景的分享

1、为什么Spring和IDEA 都不推荐使用 Autowired 注解 大家在使用IDEA开发的时候有没有注意到过一个提示&#xff0c;在字段上使用Spring的依赖注入注解Autowired后会出现如下警告Field injection is not recommended (字段注入是不被推荐的)&#xff1b;但是使用Resource却不会…

Python基础详解二

一&#xff0c;函数 函数是组织好的&#xff0c;可重复使用的&#xff0c;用来实现某个功能的代码段 def myMethod(data):print("数据长度为",len(data))myMethod("dsdsdsds") 函数的定义&#xff1a; def 函数名(传入参数):函数体return 返回值 def m…

10_Linux中的计划任务

10_Linux中的计划任务 常见计划任务 Linux系统中默认在执行的计划任务 日志文件的轮询:logrotate日志文件分析logwatch任务建立locate数据库建立manpage查询数据库RPM软件日志文件删除缓存与网络有关的分析 仅执行一次的计划任务 atd和at [rootnode4 ~]# systemctl start…

关系型数据库MySQL开发要点之多表设计案例详解代码实现

什么是多表设计 项目开发中 在进行数据库表结构设计时 根据数据模型和业务关系 会根据业务需求和业务模块之间的关系分析设计表结构 由于业务之间互相关联 所以表结构之间也存在着各种联系 主要分为以下三种 一对多 每个部门下是有多个员工的 但是一个员工只能归属一个部…

CMakeLists.txt语法规则:foreach 循环基本用法

一. 简介 cmake 中除了 if 条件判断之外&#xff0c;还支持循环语句&#xff0c;包括 foreach()循环、while()循环。 本文学习 CMakeLists.txt语法中的循环语句。 CMakeLists.txt语法中 有两种 循环实现方式&#xff1a;foreach循环与 while循环。 二. CMakeLists.txt语法规则…

设计列表结构

实现1.1&#xff0c;1.2的有序列表 <!DOCTYPE html> <html><head><meta charset"utf-8"><title></title><style>ol{list-style: none;}li:before{color:#f00;font-family:Times New Roman;}li{counter-increment:a 1;}li…

免费https证书申请

HTTPS证书&#xff0c;也称为SSL证书&#xff08;Secure Sockets Layer&#xff09;或TLS证书&#xff08;Transport Layer Security&#xff09;&#xff0c;是一种数字证书&#xff0c;用于在互联网通信中确保数据传输的安全性、完整性和真实性。它是基于公钥基础设施&#x…

RISCV 外部GCC 工具链安装@FreeBSD15

在交叉编译的时候&#xff0c;可以使用FreeBSD15默认的工具链&#xff1a;LLVM 也可以使用GCC工具链&#xff0c;GCC可以使用现成pkg包安装&#xff0c;也可以编译安装。 LLVM的特点是高移植性和高效&#xff0c;但学习成本高。GCC的特点是成熟稳定&#xff0c;但优化能力有限…

Qt之摄像头操作

简单的摄像头测试类 头文件camerawidget.h #ifndef CAMERAWIDGET_H #define CAMERAWIDGET_H#include <QWidget> #include <QList> #include <QCamera> #include <QCameraInfo> #include <QCameraViewfinder> #include <QCameraImageCapture…

基于FPGA的数字电子钟VHDL代码Quartus仿真

名称&#xff1a;基于FPGA的数字电子钟VHDL代码Quartus仿真&#xff08;文末获取&#xff09; 软件&#xff1a;Quartus 语言&#xff1a;VHDL 代码功能&#xff1a; 数字电子钟 1)设计一个能显示秒、分、时的24小时数字钟 2)用数码管显示出时&#xff0c;分&#xff0c;…

OpenSBI 固件代码分析合集-泰晓社区

泰晓社区&#xff1a;https://tinylab.org/ OpenSBI 固件代码分析&#xff08;一&#xff09;&#xff1a;启动流程OpenSBI 固件代码分析&#xff08;二&#xff09;&#xff1a;fw_base.S 源码分析OpenSBI 固件代码分析&#xff08;三&#xff09;: sbi_init.cOpenSBI 固件代…

JDK生成https配置

keytool -genkey -v -alias tomcat -keyalg RSA -keystore D:\https证书\weChat.keystore -validity 36500 -keypass 250250 keytool -importkeystore -srcstoretype JKS -srckeystore D:\https证书\weChat.keystore -srcstorepass 250250 -srcalias tomcat -srckeypass 25025…

Unity射击游戏开发教程:(10)创建主界面

主界面开发 玩游戏时,主菜单是事后才想到要做的。实际上几乎每个游戏都有一个主界面。如果你点击打开游戏并立即开始游戏,你会感到非常惊讶。本文将讨论如何创建带有启动新游戏的交互式按钮的主界面/主菜单。 主菜单将是一个全新的场景。我们将添加一个 UI 图像元素,并在图像…

NLP(13)--文本分类任务

前言 仅记录学习过程&#xff0c;有问题欢迎讨论 情感分析&#xff0c;违规检测&#xff0c;商品评论打分 贝叶斯算法&#xff1a; P(B1) 结果为奇数 P(B2) 结果为偶数 P(A) 结果为5 P(A) P(B1) * P(A|B1) P(B2) * P(A|B2) 1/2 1/3 1/20 支持向量机&#xff1a;les…

java中的变量、数据类型、人机交互

变量 变量要素 1、类型&#xff1b;每一个变量都需要定义类型&#xff08;强类型&#xff09;其它语言有弱类型&#xff08;js&#xff09; 2、变量名&#xff1b; 3、存储的值&#xff1b; 声明方式&#xff1a; 数据类型 变量名 变量值&#xff1b; public static vo…

Java之抽象类和接口

一、抽象类 1.抽象类概念 如果一个类中没有包含足够的信息来描绘一个具体的对象&#xff0c;这样的类就是抽象类&#xff0c; 比如动物类。没有实际工作的方法 , 我们可以把它设计成一个 抽象方法&#xff0c; 包含抽象方法的类我们称为 抽象类。 2.抽象类语法 在Java中&am…

网络基础——校验

网络基础——校验 网络通信的层次化模型&#xff08;如OSI七层模型或TCP/IP四层模型&#xff09;中&#xff0c;每一层都有其特定的校验机制来确保数据传输的正确性和完整性。 物理层 校验方式 不直接涉及校验和&#xff0c;但会采用信号编码技术&#xff08;如曼彻斯特编码…

Linux磁盘IO、网络IO、零拷贝详解

一、什么是I/O&#xff1f; 在计算机操作系统中&#xff0c;所谓的I/O就是输入&#xff08;input&#xff09;和输出&#xff08;output&#xff09;,也可以理解为读&#xff08;read&#xff09;和写&#xff08;write&#xff09;,针对不同的对象&#xff0c;I/O模式可以划分…

什么是接口和类?Java中的集合框架有哪些主要接口和类?

Java中的集合框架有哪些主要接口和类&#xff1f; Java中的集合框架&#xff08;Java Collections Framework&#xff09;提供了一套丰富的接口和类&#xff0c;用于存储和操作对象的集合。以下是Java集合框架中的主要接口和类&#xff1a; 主要接口 Collection&#xff1a; 这…

【busybox记录】【shell指令】comm

目录 内容来源&#xff1a; 【GUN】【comm】指令介绍 【busybox】【comm】指令介绍 【linux】【comm】指令介绍 使用示例&#xff1a; 逐行比较两个排序后的文件 - 默认输出 逐行比较两个排序后的文件 - 如果一个文件的排序有问题&#xff0c;那么反错&#xff08;默认&…