【进程终止】退出信号 | 三种退出情况 | 如何进程终止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函数终止进程。
//先冲刷缓冲区再休眠/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.mfbz.cn/a/599530.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

批量将GOID转成GO term名并添加BP,MF,CC分类信息

基因本体论&#xff08;Gene Ontology&#xff0c;GO&#xff0c;https://www.geneontology.org&#xff09;是一个广泛应用于生物信息学领域的知识库&#xff0c;它提供了一套标准化的词汇和分类体系&#xff0c;用于描述基因功能、细胞组分和生物过程。GO旨在统一科研人员对基…

C/C++ BM30 二叉搜索树与双向链表

文章目录 前言题目解决方案一1.1 思路阐述1.2 源码 解决方案二2.1 思路阐述2.2 源码 总结 前言 这道题要明白二叉搜索树的概念&#xff0c;同时还要对链表的知识比较熟悉。 题目 输入一棵二叉搜索树&#xff0c;将该二叉搜索树转换成一个排序的双向链表。如下图所示 数据范…

在QEMU上运行OpenSBI+Linux+Rootfs

在QEMU上运行OpenSBILinuxRootfs 1 编译QEMU2 安装交叉编译工具3 编译OpenSBI4 编译Linux5 创建根文件系统5.1 编译busybox5.2 创建目录结构5.3 制作文件系统镜像5.3.1 创建 ext2 文件5.3.2 将目录结构拷贝进 ext2 文件5.3.3 取消挂载 6 运行OpenSBILinuxRootfs 本文所使用的版…

PVZ2 植物克僵尸【第二期】

众所周知&#xff0c;PVZ2&#xff08;植物大战僵尸2&#xff09;中有许多恶心的僵尸&#xff0c;而我们不得不派出它们的————克星&#xff01;&#xff08;*为建议方法&#xff09; 5.战机小鬼 战机小鬼&#xff0c;恶心会发射子弹&#xff0c;所以&#xff1a; 1&…

(三)JSP教程——JSP动作标签

JSP动作标签 用户可以使用JSP动作标签向当前输出流输出数据&#xff0c;进行页面定向&#xff0c;也可以通过动作标签使用、修改和创建对象。 <jsp:include>标签 <jsp:include>标签将同一个Web应用中静态或动态资源包含到当前页面中。资源可以是HTML、JSP页面和文…

【解决】:git clone项目报错fatal: fetch-pack: invalid index-pack output

象&#xff1a;之前一直使用gitee将个人学习和工作相关记录上传到个人gitee仓库&#xff0c;一直没出现过问题。直到有一天换电脑重新拉取代码发现出了问题&#xff0c;具体如下图&#xff1a; 原因分析&#xff1a; 经过查询发现主要原因是因为git clone的远程仓库的项目过大…

【强化学习】公平性Actor-Critic算法

Bringing Fairness to Actor-Critic Reinforcement Learning for Network Utility Optimization 阅读笔记 Problem FormulationLearning AlgorithmLearning with Multiplicative-Adjusted RewardsSolving Fairness Utility Optimization Evaluations 在网络优化问题中&#xff…

懒人网址导航源码v3.9源码及教程

懒人网址导航源码v3.9源码及教程 效果图使用方法部分源码领取源码下期更新预报 效果图 使用方法 测试环境 宝塔Nginx -Tengine2.2.3的PHP5.6 MySQL5.6.44为防止调试错误&#xff0c;建议使用测试环境运行的php与mysql版本首先用phpMyAdmin导入数据库文件db/db.sql 如果导入不…

QT-TCP通信

网上的资料太过于书面化&#xff0c;所以看起来有的让人云里雾里&#xff0c;看不懂C-tcpsockt和S-tcpsocket的关系 所以我稍微画了一下草图帮助大家理解两个套接字之间的关系。字迹有的飘逸勉强看看 下面是代码 服务端&#xff1a; MainWindow::MainWindow(QWidget *parent) …

Kubernetes 教程:在 Containerd 容器中使用 GPU

原文链接:Kubernetes 教程:在 Containerd 容器中使用 GPU 云原生实验室本文介绍了如何在使用 Containerd 作为运行时的 Kubernetes 集群中使用 GPU 资源。https://fuckcloudnative.io/posts/add-nvidia-gpu-support-to-k8s-with-containerd/ 前两天闹得沸沸扬扬的事件不知道…

Golang | Leetcode Golang题解之第67题二进制求和

题目&#xff1a; 题解&#xff1a; func addBinary(a string, b string) string {ans : ""carry : 0lenA, lenB : len(a), len(b)n : max(lenA, lenB)for i : 0; i < n; i {if i < lenA {carry int(a[lenA-i-1] - 0)}if i < lenB {carry int(b[lenB-i-1…

6W 1.5KVDC. 单、双输出 DC/DC 电源模块——TP2L-6W 系列

TP2L-6W系列是一款高性能、超小型的电源模块&#xff0c;2:1电压输入&#xff0c;输出有稳压和连续短路保护功能&#xff0c;隔离电压为1.5KVDC、作温度范围为–40℃到85℃。特别适合对输出电压的精度有严格要求的地方&#xff0c;外部遥控功能对您的设计又多一项选择&#xff…

Liunx磁盘管理(中)

Liunx磁盘管理(上)-CSDN博客 目录 查看块设备信息 lsblk&#xff08;list block devices&#xff09; fdisk gdisk parted blkid df&#xff08;disk free&#xff09; 虚拟机添加硬盘 步骤&#xff1a; 磁盘分区 MBR格式创建分区 使用方法 替代工具 GPT分区格式…

【C 数据结构-动态内存管理】2. 边界标识法管理动态内存

文章目录 【 1. 边界标识法的结构设计 】【 2. 分配算法 】【 3. 回收算法 】3.1 空闲块两侧是占用块3.2 空闲块左侧是空闲块3.3 空闲块右侧是空闲块3.3 空闲块两侧是空闲块 边界标识法 可以解决系统中内存碎片过多而无法使用的问题。 【 1. 边界标识法的结构设计 】 使用边界…

绘唐ai工具怎么获取

这款产品的最大亮点在于其高度精准的语音克隆能力&#xff0c;利用先进的模型&#xff0c;能够捕捉到用户独特的音调、音高和调制方式&#xff0c;使用户能够以前所未有的方式复制和利用自己的声音。仅需10秒钟的录制时间&#xff0c;即可实现声音的克隆&#xff0c;相当便捷。…

Spring Cloud 整合Sentinel

1、引入依赖 版本说明 alibaba/spring-cloud-alibaba Wiki GitHub 父pom <spring.cloud.version>Hoxton.SR12</spring.cloud.version> <spring.cloud.alibaba.version>2.2.10-RC1</spring.cloud.alibaba.version>Sentinel应用直接引用starter <…

基于FPGA的数字密码锁电路Verilog代码Quartus仿真

名称&#xff1a;基于FPGA的数字密码锁电路Verilog代码Quartus仿真(文末获取&#xff09; 软件&#xff1a;Quartus 语言&#xff1a;Verilog 代码功能&#xff1a; 数字密码锁电路的设计 1.设计任务:设计并制作数字密码锁电路 2.设计要求 1.用EDA实训仪的I/设备和PLD志…

纯血鸿蒙APP实战开发——折叠屏扫描二维码方案

折叠屏扫描二维码方案 介绍 本示例介绍使用自定义界面扫码能力在折叠屏设备中实现折叠态切换适配。自定义界面扫码使用系统能力customScan&#xff0c;其提供相机流的初始化、启动扫码、识别、停止扫码、释放相机流资源等能力。折叠屏折叠状态通过监听display的foldStatusCha…

Hive3.0新特性:Materialized Views 物化视图

Materialized Views 物化视图 在 Apache Hive 3.0 中引入了物化视图&#xff08;Materialized Views&#xff09;的支持&#xff0c;它们是预先计算并缓存了查询结果的数据结构&#xff0c;以提高查询性能和降低延迟。物化视图通过将查询的结果存储在物理表中来实现&#xff0…

深入理解指针1

目录 如对您有帮助&#xff0c;还望三连支持&#xff0c;谢谢&#xff01;&#xff01;&#xff01; 1.内存和地址 计算机中常⻅的单位&#xff08;补充&#xff09;&#xff1a; 如何理解编址 2.指针变量和地址 2.1取地址操作符&#xff08;&&#xff09; 2.2指针变…
最新文章