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

news2025/1/15 12:46:36

目录

退出码

退出信号

进程终止情况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函数终止进程。
//先冲刷缓冲区再休眠/n
  1 #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没有刷新缓冲区的数据,就直接终止程序了。

//先冲刷缓冲区再休眠/n
  1 #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.coloradmin.cn/o/1650361.html

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

相关文章

选择器、pxcook软件、盒子模型

结构伪类选择器 定义&#xff1a;根据结构的元素关系来查找元素。 <title>Document</title><style>li:first-child{color:aqua ;}li:last-child{color: aqua;}li:nth-child(3){color: aqua;}</style> </head> <body><ul><li>…

端口被其他进程占用:OSError: [Errno 98] Address already in use

一、问题描述 错误提示端口号正在被使用 二、解决办法 1.使用 lsof 命令&#xff0c;列出所有正在监听&#xff08;即被绑定&#xff09;的网络连接&#xff0c;包括它们所使用的端口号 sudo lsof -i -P -n | grep LISTEN 2.解绑被绑定的端口号 根据 netstat 或 lsof 命令…

C#修改默认参数settings文件

右击项目在设置中进行修改&#xff1a; 千万不要在这里改。 如果要在自己的项目里添加这个文件&#xff0c;首先新建个文件夹&#xff0c;然后添加.setting文件&#xff0c;然后再像上面说的那样添加属性。

Ansible自动化工具模块调用与playbook编写

目录 一、Ansible工作机制与特点 &#xff08;一&#xff09;Ansible工作机制 1. 初始化与配置 2. 编写Playbook 3. 调用模块 4. 加密敏感数据 5. 执行Playbook 6. 收集执行结果 7. 错误处理与回滚 8. 反馈与报告 &#xff08;二&#xff09;Ansible 的主要特点包括…

BUUCTF [极客大挑战 2019]EasySQL 1

BUUCTF:https://buuoj.cn/challenges 题目描述&#xff1a; [极客大挑战 2019]EasySQL 1 密文&#xff1a; 解题思路&#xff1a; 1、根据题目提示&#xff0c;并且网站也存在输入框&#xff0c;尝试进行SQL注入。 首先&#xff0c;判断提交方式&#xff0c;随机输入数据…

力扣刷题--数组--第二天

今天仍然做二分查找相关的题目。先来回顾一下二分查找的方法和使用的条件。二分查找是在数组中查找目标值的一种方法&#xff0c;通过边界索引确定中间索引&#xff0c;判断中间索引处的元素值和目标值的大小&#xff0c;来不断缩小查找区间。使用二分查找有如下一些限制&#…

PXE批量安装

系统装机的三种引导方式 u盘光盘网络装机 光盘&#xff1a; 1.类似于usb模式 2.刻录模式 系统安装过程 加载boot loader Boot Loader 是在操作系统内核运行之前运行的一段小程序。通过这段小程序&#xff0c;我们可以初始化硬件设备、建立内存空间的映射图&#xff0c;从…

使用开放式用户通信连接两台西门子S71200plc

步骤1.在项目中创建两台PLC。 步骤2.分别设置两个PLC的参数。 plc1 plc2 步骤3.对两个plc进行组态 步骤4.在plc1和plc2中各自创建DB块&#xff0c;用于通信。 须在块的属性中取消优化块的访问选项。 plc1 plc2 步骤5.往plc1的main块中编写代码。 步骤6.往plc2的main块中编写…

AndroidStudio的Iguana版的使用

1.AndroidStudio介绍 Android Studio 是用于开发 Android 应用的官方集成开发环境 (IDE)。Android Studio 基于 IntelliJ IDEA 强大的代码编辑器和开发者工具&#xff0c;还提供更多可提高 Android 应用构建效率的功能&#xff0c;例如&#xff1a; 基于 Gradle 的灵活构建系统…

XORM 框架的使用

1、xorm 1.1、xorm 简介 xorm 是一个简单而强大的Go语言ORM库. 通过它可以使数据库操作非常简便。 特性 支持 struct 和数据库表之间的灵活映射&#xff0c;并支持自动同步事务支持同时支持原始SQL语句和ORM操作的混合执行使用连写来简化调用支持使用ID, In, Where, Limit,…

nginx模型设计和进程讲解

一. Nginx进程模型解析 1. master主进程 和 worker工作进程 [rootlocalhost sbin]# ps -ef|grep nginx root 15411 1 0 21:08 ? 00:00:00 nginx: master process ./nginx nobody 15412 15411 0 21:08 ? 00:00:00 nginx: worker process root…

pytest教程-39-钩子函数-pytest_runtest_setup

领取资料&#xff0c;咨询答疑&#xff0c;请➕wei: June__Go 上一小节我们学习了pytest_runtest_protocol钩子函数的使用方法&#xff0c;本小节我们讲解一下pytest_runtest_setup钩子函数的使用方法。 pytest_runtest_setup 钩子函数在每个测试用例的 setup 阶段被调用。这…

学习软考----数据库系统工程师22

关系运算 基本的关系代数运算 拓展的关系运算 除&#xff1a;需要S连接中属性为C和D的两个元组都与R连接一样&#xff0c;且在R连接中对应的另外的元素完全一致 总结

聊聊 ASP.NET Core 中间件(三):如何创建自己的中间件?

前言 本质上&#xff0c;中间件类也是一个普通的 .NET 类&#xff0c;它不需要继承任何父类或者实现任何接口。 但是有几个约定&#xff1a; 需要有一个构造方法构造方法至少要有一个 RequestDelegate 类型的参数&#xff0c;用来指向下一个中间件。需要定义一个名字为 Invo…

交易复盘-20240507

仅用于记录当天的市场情况&#xff0c;用于统计交易策略的适用情况&#xff0c;以便程序回测 短线核心&#xff1a;不参与任何级别的调整&#xff0c;采用龙空龙模式 一支股票 10%的时候可以操作&#xff0c; 90%的时间适合空仓等待 蔚蓝生物 (5)|[9:25]|[36187万]|4.86 百合花…

STM32F10x移植FreeRTOS

一、获取FreeRTOS源码 &#xff08;1&#xff09;登录FreeRTOS官网&#xff1a;www.freertos.org&#xff0c;下载第一个压缩包 &#xff08;2&#xff09;通过GitHub网站&#xff1a;github.com/FreeRTOS/FreeRTOS下载&#xff0c;由于该网站服务器在国外&#xff0c;所以访问…

28.leetcode---前K个高频单词(Java版)

题目链接: https://leetcode.cn/problems/top-k-frequent-words/description/ 题解: 代码: 测试:

巨控GRM561/562/563/564Q杀菌信息远程监控

摘要 通过程序编写、手机APP画面制作等运行系统&#xff0c;实现电脑及手机APP显示的历史曲线画面和数据图形化的实时性。 不仅流程效率提升90%以上&#xff0c;同时为杀菌生产提供有利的质量保障&#xff0c;还有效规避因触屏及内存卡的突发异常导致历史数据的丢失&#xff0…

强大的禄得可转债自定义因子轮动系统完成,可转债三低为例子

经过几天的测试终于完成了可转债自定义因子轮动&#xff0c;超过1000行的源代码 我提供了服务器的数据支持自动api下载&#xff0c;我给大家维护数据 网页 http://120.78.132.143:8023/ 录得数据支持http://120.78.132.143:8023/lude_data_app api数据支持&#xff0c;我提供…

杭州恒生面试,社招,3年经验

你好&#xff0c;我是田哥 一位朋友节前去恒生面试&#xff0c;其实面试问题大部分都是八股文&#xff0c;但由于自己平时工作比较忙&#xff0c;完全没有时间没有精力去看八股文&#xff0c;导致面试结果不太理想&#xff0c;HR说节后通知面试结果&#xff08;估计是凉了&…