【进程终止】退出信号 | 三种退出情况 | 如何进程终止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://xiahunao.cn/news/3016511.html

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

相关文章

springboot项目组合定时器schedule注解实现定时任务

springboot项目组合定时器schedule注解实现定时任务&#xff01; 创建好springboot项目后&#xff0c;需要在启动类上增加注解开启定时器任务 下图所示&#xff1a; 增加这个注解&#xff0c;启动项目&#xff0c; package com.example.scheduledemo.util;import org.springf…

Linux进程通信-信号

信号概念 信号是 Linux 进程间通信的最古老的方式之一&#xff0c;是事件发生时对进程的通知机制&#xff0c;有时也称之为软件中断&#xff0c;它是在软件层次上对中断机制的一种模拟&#xff0c;是一种异步通信的方式。信号 可以导致一个正在运行的进程被另一个正在运行的异…

企业怎样进行IT外包以及IT外包服务内容

在数字化时代的浪潮中&#xff0c;企业逐渐认识到信息技术的关键作用&#xff0c;特别是制造业基地对于IT外包和运维服务的需求持续增长。然而&#xff0c;在诸多可供选择的IT外包和运维方案中&#xff0c;企业如何推动与IT外包公司的合作&#xff1f;本文将深入介绍IT外包方案…

nginx 启动,查看,停止

nginx 启动&#xff0c;查看&#xff0c;停止 启动 start nginx 查看是否启动成功 tasklist | findstr nginx 停止 nginx -s stop 测试配置文件的语法是否有误 nginx -t 重启nginx nginx-s reload

网络安全之动态路由OSPF基础

OSPF&#xff1a;开放式最短路径优先协议。 1、协议使用范围&#xff1a;IGP。 2、协议算法特点&#xff1a;链路状态型路由协议。 3、协议是否传递网络掩码&#xff1a;传递网络掩码&#xff08;无类别的路由协议&#xff09;。 4、协议封装&#xff1a;基于IP协议封装&am…

第六代移动通信介绍、无线网络类型、白皮书

关于6G 即第六代移动通信的介绍&#xff0c; 图解通信原理与案例分析-30&#xff1a;6G-天地互联、陆海空一体、全空间覆盖的超宽带移动通信系统_6g原理-CSDN博客文章浏览阅读1.7w次&#xff0c;点赞34次&#xff0c;收藏165次。6G 即第六代移动通信&#xff0c;6G 将在5G 的基…

《QT实用小工具·六十》Qt 多列时间轴控件

1、概述 源码放在文章末尾 Qt 多列时间轴控件。 可与多段字符串格式自由转换&#xff0c;也可手动添加列表项。 专门用来以时间轴作为事件线发展顺序的故事大纲。 特点 时间背包功能&#xff1a;记录所有物品或属性发生的变化&#xff0c;随时回溯 时间可输入任意内容&…

【区块链】智能合约简介

智能合约起源 智能合约这个术语至少可以追溯到1995年&#xff0c;是由多产的跨领域法律学者尼克萨博&#xff08;NickSzabo&#xff09;提出来的。他在发表在自己的网站的几篇文章中提到了智能合约的理念。他的定义如下&#xff1a;“一个智能合约是一套以数字形式定义的承诺&a…

【C++STL详解(八)】--------stack和queue的模拟实现

目录 前言 一、stack模拟实现 二、queue的模拟实现 前言 前面也介绍了stack和queue的常见接口&#xff0c;我们也知道stack和queue实际上是一种容器适配器&#xff0c;它们只不过是对底层容器的接口进行封装而已&#xff0c;所以模拟实现起来比较简单&#xff01;一起来看看是…

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

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

北邮22级信通院DSP:实验三(1):FFT变换、IFFT变换(附每步8点变换蝶形图)保姆级讲解+用C++程序实现复数域的FFT变换和IFFT变换

北邮22信通一枚~ 跟随课程进度更新北邮信通院DSP的笔记、代码和文章&#xff0c;欢迎关注~ 获取更多文章&#xff0c;请访问专栏&#xff1a; 北邮22级信通院DSP_青山入墨雨如画的博客-CSDN博客 目录 一、预备知识 1.1 FFT算法 1.2.1由DFT到FFT 1.2.2 基2时域抽选算法 …

华为:三层交换机与路由器连通上网实验

三层交换机是一种网络交换机&#xff0c;可以实现基于IP地址的高效数据转发和路由功能&#xff0c;通常用于大型企业、数据中心和校园网络等场景。此外&#xff0c;三层交换机还支持多种路由协议&#xff08;如OSPF、BGP等&#xff09;&#xff0c;以实现更为复杂的网络拓扑结构…

重装win11系统后找不到WiFi

由于电脑崩溃重装了系统&#xff0c;win11,装完之后WiFi图标不见了且网络适配器根本没有无线网络选项。 右键电脑》管理》网络适配器。 在刚装好系统时候并没有前两项&#xff0c;查了很多资料&#xff0c;比如 关机14s 重启&#xff0c;还有通过服务配置 WLAN AutoConfig 都…

【嵌入式DIY实例】-基于GSM的远程灌溉系统

基于GSM的远程灌溉系统 文章目录 基于GSM的远程灌溉系统1、硬件准备与接线2、软件准备3、代码实现本文将详细介绍如何搭建通过使用手机实现对灌溉系统的远程控制。该系统利用全球移动通信系统(GSM)技术在灌溉系统和移动电话之间建立通信。该系统建立在流行的开源微控制器平台…

IP规划案例

整个OSPF环境IP基于172.16.0.0/16划分 172.16.0.0/16 先分成2个网段&#xff08;OSPF RIP&#xff09;&#xff0c;借1位172.16.0.0/17 ---OSPF 再按区域划分&#xff08;5个区域&#xff09;&#xff0c;借3位 172.16.0.0/20 ---Area 0 三个环回 MGRE 4个网…

Vulnhub项目:NAPPING: 1.0.1

1、靶机介绍 靶机地址&#xff1a;Napping: 1.0.1 ~ VulnHub 2、渗透过程 老规矩&#xff0c;先探测&#xff0c;靶机ip&#xff1a;192.168.56.152 本机ip&#xff1a;192.168.56.146 来看一看靶机开放哪些端口&#xff0c;nmap一下 nmap -sS -sV -A -T5 192.168.56.152 开…

软件系统测试方案书(测试计划-Word原件)

2 引言 2.1 编写目的 2.3 测试人员 2.4 项目背景 2.5 测试目标 2.6 简写和缩略词 2.7 参考资料 2.8 测试提交文档 2.9 测试进度 3 测试环境 3.1 软硬件环境 4 测试工具 5 测试策略 5.1 测试阶段划分及内容 5.1.1 集成测试 5.1.2 系统测试 5.1.2.1 功能测试 5.…

PXE批量安装

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

「C++ 内存管理篇 00」指针

目录 一、变量&#xff0c;变量名和指针 1. 什么是变量&#xff1f; 2. 变量名和指针 3. 使用指针获取数据 二、指针变量和数组变量 三、编译器对指针的等级有着严格的检查 四、指针的加减 1. 存放指针的变量的加减 2. 存放指针的变量的自增自减 3. 两个指针相减 一、变量&…

网络基础——路由

网络基础——路由 要想网络畅通&#xff0c;应让网络中的路由器知道如何转发数据包到各个网段。路由器根据路由表来转发数据包&#xff0c;而路由表是通过直连网络、静态路由以及动态路由来构建的。 route命令&#xff0c;底层是使用ioctl实现&#xff1b;ip命令&#xff0c;…