C语言详解分析进程控制中进程终止的实现

目录
  • 进程退出的形式
  • 进程退出的几种方法

进程退出的形式

进程退出的几种情况

  • 正常退出(自愿,代码运行完其结果正确)
  • 错误退出(自愿,代码运行完其结果不正确)
  • 异常退出(非自愿,代码异常直接终止)
  • 被其他进程终止(非自愿)

自愿退出会返回一个退出码,由父进程接收。

在Linux上可以使用命令echo $?显示最近一次退出的进程返回的退出码

  //现有如下代码,源文件名为mycode.c
 # include <stdio.h>
 int main(void)
 {
   printf("i am testing\n");
   return 0;
 }

运行以上代码……

使用命令echo $?显示退出码

正常退出的退出码为0,错误退出的退出码为非0

退出码反应了进程退出的原因。所以有必要知道这些退出码究竟代表着什么。

strerror()函数的作用就是以字符的形式返回退出码。

//所在头文件
<string.h>
//函数原型
char* strerror(int errNum);
//可以使用下面代码查看退出码对应的信息
 # include <stdio.h>
 # include <string.h>
 int main(void)
 {
   for(int i = 0; i < 135; i++)
   {
     printf("%d: %s\n", i, strerror(i));
   }
   return 0;
 }

Linux系统上一共有134个退出码,退出码“134”就已经不认识它了。

0: Success
1: Operation not permitted
2: No such file or directory
3: No such process
4: Interrupted system call
5: Input/output error
6: No such device or address
7: Argument list too long
8: Exec format error
9: Bad file descriptor
10: No child processes
11: Resource temporarily unavailable
12: Cannot allocate memory
13: Permission denied
14: Bad address
15: Block device required
16: Device or resource busy
17: File exists
18: Invalid cross-device link
19: No such device
20: Not a directory
21: Is a directory
22: Invalid argument
23: Too many open files in system
24: Too many open files
25: Inappropriate ioctl for device
26: Text file busy
27: File too large
28: No space left on device
29: Illegal seek
30: Read-only file system
31: Too many links
32: Broken pipe
33: Numerical argument out of domain
34: Numerical result out of range
35: Resource deadlock avoided
36: File name too long
37: No locks available
38: Function not implemented
39: Directory not empty
40: Too many levels of symbolic links
41: Unknown error 41
42: No message of desired type
43: Identifier removed
44: Channel number out of range
45: Level 2 not synchronized
46: Level 3 halted
47: Level 3 reset
48: Link number out of range
49: Protocol driver not attached
50: No CSI structure available
51: Level 2 halted
52: Invalid exchange
53: Invalid request descriptor
54: Exchange full
55: No anode
56: Invalid request code
57: Invalid slot
58: Unknown error 58
59: Bad font file format
60: Device not a stream
61: No data available
62: Timer expired
63: Out of streams resources
64: Machine is not on the network
65: Package not installed
66: Object is remote
67: Link has been severed
68: Advertise error
69: Srmount error
70: Communication error on send
71: Protocol error
72: Multihop attempted
73: RFS specific error
74: Bad message
75: Value too large for defined data type
76: Name not unique on network
77: File descriptor in bad state
78: Remote address changed
79: Can not access a needed shared library
80: Accessing a corrupted shared library
81: .lib section in a.out corrupted
82: Attempting to link in too many shared libraries
83: Cannot exec a shared library directly
84: Invalid or incomplete multibyte or wide character
85: Interrupted system call should be restarted
86: Streams pipe error
87: Too many users
88: Socket operation on non-socket
89: Destination address required
90: Message too long
91: Protocol wrong type for socket
92: Protocol not available
93: Protocol not supported
94: Socket type not supported
95: Operation not supported
96: Protocol family not supported
97: Address family not supported by protocol
98: Address already in use
99: Cannot assign requested address
100: Network is down
101: Network is unreachable
102: Network dropped connection on reset
103: Software caused connection abort
104: Connection reset by peer
105: No buffer space available
106: Transport endpoint is already connected
107: Transport endpoint is not connected
108: Cannot send after transport endpoint shutdown
109: Too many references: cannot splice
110: Connection timed out
111: Connection refused
112: Host is down
113: No route to host
114: Operation already in progress
115: Operation now in progress
116: Stale file handle
117: Structure needs cleaning
118: Not a XENIX named type file
119: No XENIX semaphores available
120: Is a named type file
121: Remote I/O error
122: Disk quota exceeded
123: No medium found
124: Wrong medium type
125: Operation canceled
126: Required key not available
127: Key has expired
128: Key has been revoked
129: Key was rejected by service
130: Owner died
131: State not recoverable
132: Operation not possible due to RF-kill
133: Memory page has hardware error
134: Unknown error 134

Linux上一切皆为文件,指令也是一个文件,当使用不存在的指令时,就会出现报错信息。

其退出码为2,退出码2代表的含义为: No such file or directory(没有这样的文件或目录)

进程退出的几种方法

正常终止的3种方法:

  • main函数返回
  • 调用exit
  • 调用_exit

最常见的进程终止莫过于main函数返回了!需要注意main函数中返回才是进程退出,而非main函数返回并不能正常终止进程,只是函数返回。

【调用exit】

//所在头文件
 #include <stdlib.h>
 //exit函数原型
 void exit(int status);

exit函数是语言层面的函数,是对系统调用函数_exit的封装。

它可以在main函数中调用,也可以在非main函数中调用,都可以达到终止进程的目的。

C标准还规定了两个宏EXIT_SUCCESS和EXIT_FAILURE可以作为exit的参数,用来指示这个进程是正常退出还是错误退出。

 # include <stdio.h>
 # include <string.h>
 # include <stdlib.h>
 int main(void)
 {
   printf("i am testing\n");
   exit(EXIT_SUCCESS);
   return 0;
 }

 # include <stdio.h>
 # include <string.h>
 # include <stdlib.h>
 int main(void)
 {
   printf("i am testing\n");
   exit(EXIT_FAILURE);     //修改exit函数的参数
   return 0;
 }

【调用_exit函数】

//所在头文件
#include <unistd.h>
//_exit函数原型
void _exit(int status);

_exit是系统调用函数。_exit函数的本质就是调用了系统调用函数_exit。

exit函数封装了系统调用_exit,exit函数还做了其他的事情。

 # include <stdio.h>
 # include <string.h>
 # include <stdlib.h>
 # include <unistd.h>
 int main(void)
 {
   printf("i am testing"); //没有“\n”
   sleep(2);
   exit(EXIT_SUCCESS);
   return 0;
 }

代码运行的现象:休眠2秒后,在屏幕上打印“i am testing”.

因此,可以看出,exit函数会冲刷缓冲区……

现在换做_exit函数来终止进程。

 # include <stdio.h>
 # include <string.h>
 # include <stdlib.h>
 # include <unistd.h>
 int main(void)
 {
   printf("i am testing"); //没有“\n”
   sleep(2);
   _exit(EXIT_SUCCESS);
   return 0;
 }

代码运行的现象:休眠两秒后,并没有在屏幕上显示预期的“i am testing”,就终止进程了。

因此可看出,_exit函数并不会冲刷缓冲区……

实际上,_exit系统调用函数只是简单地终止了进程,并不会做其他的工作。而exit函数在调用_exit函数之前还会做一些其他的工作:

  • 执行用户定义的清理函数
  • 冲刷缓冲区,关闭所有的流等。
  • 调用_exit函数

main函数返回,本质上是把返回值作为参数传给exit函数,最后还是调用exit函数。

到此这篇关于C语言详解分析进程控制中进程终止的实现的文章就介绍到这了,更多相关C语言进程终止内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C语言控制进程之进程等待详解

    目录 进程等待的必要 进程等待的方法 wait函数 waitpid函数 获取子进程退出信息 进程等待的必要 当一个进程终止的时候,它的资源,比如说PCB,数据等不会被立马清理掉.它会保持在已经终止的状态,这种状态称为“僵尸状态”,直到被父进程确认.父进程wait,即父进程向内核确认子进程已经终止,可以为子进程“收尸”了,内核会把子进程的退出信息传给父进程,然后清理掉子进程的资源,这个时候子进程才算真正地终止了! 总结: 父进程等待,可以获取子进程的退出信息,知道子进程的执行结果. 父进程等待,可

  • C语言详解分析进程控制中进程终止的实现

    目录 进程退出的形式 进程退出的几种方法 进程退出的形式 进程退出的几种情况 正常退出(自愿,代码运行完其结果正确) 错误退出(自愿,代码运行完其结果不正确) 异常退出(非自愿,代码异常直接终止) 被其他进程终止(非自愿) 自愿退出会返回一个退出码,由父进程接收. 在Linux上可以使用命令echo $?显示最近一次退出的进程返回的退出码 //现有如下代码,源文件名为mycode.c # include <stdio.h> int main(void) { printf("i am

  • C语言详解数据结构与算法中枚举和模拟及排序

    目录 枚举 连号区间数 递增三元组 二分 双指针 前缀和 模拟 特别数的和 错误票据 排序 快速排序 归并排序 枚举 连号区间数 来源:第四届蓝桥杯省赛C++B组,第四届蓝桥杯省赛JAVAB组 小明这些天一直在思考这样一个奇怪而有趣的问题: 在 1∼N 的某个排列中有多少个连号区间呢? 这里所说的连号区间的定义是: 如果区间 [L,R] 里的所有元素(即此排列的第 L 个到第 R 个元素)递增排序后能得到一个长度为 R−L+1 的“连续”数列,则称这个区间连号区间. 当 N 很小的时候,小明可以

  • C语言中指针和数组试题详解分析

    目录 数组题: 程序一(一维数组): 字符数组 程序二(字符数组): 程序三(字符数组): 程序四(字符数组): 程序五(字符数组): 二维数组 程序六( 二维数组): 指针题 程序七( 指针): 程序八( 指针): 程序九( 指针): 程序十( 指针): 程序十( 图): 程序十一( 指针): 程序十二( 指针): 程序十三( 指针): 指针 和 数组 试题解析 小编,在这里想说一下,c语言的最后一节 C预处理,可能还需要一些时间,因为小编,昨天才下载了虚拟机 和 linux 系统,还没开始安

  • C语言 分支语句详解分析

    目录 分支语句的分类 1.单分支结构语法形式: 2.双分支结构 3.多分支结构 注意事项 分支语句的分类 在不同的场景,我们常常面临着选择,有时候会在爱情与面包之间选择,有时候会在尊严与生存之间选择,又或者面对琳琅满目的商品时,踌躇不定的选择.总之,选择存在于我们生活的方方面面.但是,万物皆可程序化. 分支语句我们分为三种. 单分支结构 双分支结构 多分支结构 1.单分支结构语法形式: if( 条件表达式 ) { (语句块) } 2.双分支结构 双分支结构的基本形式是: if( 条件表达式) {

  • C语言数据结构之单向链表详解分析

    链表的概念:链表是一种动态存储分布的数据结构,由若干个同一结构类型的结点依次串连而成. 链表分为单向链表和双向链表. 链表变量一般用指针head表示,用来存放链表首结点的地址. 每个结点由数据部分和下一个结点的地址部分组成,即每个结点都指向下一个结点.最后一个结点称为表尾,其下一个结点的地址部分的值为NULL(表示为空地址). 特别注意:链表中的各个结点在内存中是可以不连续存放的,具体存放位置由系统分配. 例如:int *ptr ; 因此不可以用ptr++的方式来寻找下一个结点. 使用链表的优点

  • C语言程序环境和预处理详解分析

    目录 一.程序的翻译环境和运行环境 程序的翻译环境 链接阶段 执行环境(运行环境) 二.预处理详解 预定义符号 #define定义标识符 #define定义宏 #define 替换规则 #和##两个预处理的工具 带副作用的宏参数 宏和函数对比 #undef移除宏 命令行定义 条件编译 头文件包含 嵌套文件包含 总结 一.程序的翻译环境和运行环境 重点:任何ANSI C(标准C的程序)的一种实现,存在两个不同的环境 第1种是翻译环境,在这个环境中源代码被转换为可执行的机器指令. 第2种是执行环境,

  • C语言详解判断相同树案例分析

    目录 一.题目描述 二.解题思路 题目难度:简单 一.题目描述 给你两棵二叉树的根节点 p 和 q ,编写一个函数来检验这两棵树是否相同. 如果两个树在结构上相同,并且节点具有相同的值,则认为它们是相同的. LeetCode链接:相同的树 二.解题思路 核心思路: 先比较两颗二叉树的根节点 如果「都为空」,则返回 true,说明两树相同. 如果「一个为空一个不为空」,说明这两颗树不相同,则返回 false. 如果「都不为空,但节点值不相同」,说明这两颗树不相同,则返回 false. 经过 1 和

  • C语言详解strcmp函数的分析及实现

    目录 1.函数介绍 1.1.函数接口 1.2.函数分析 1.3.函数的简单使用 1.4.函数使用结果分析 2.库函数strcmp源代码 2.1.库函数源代码 2.2.库函数分析 3.模拟实现 strcmp 函数 3.1.模拟实现 3.2.模拟实现分析 1.函数介绍 1.1.函数接口 int __cdecl strcmp (const char * src,const char * dst); 这里是库函数里面的函数定义接口.这个函数是将 src 和 dst 两个字符串进行比较,即为字符串比较函数

  • C语言 详解如何删除有序数组中的重复项

    目录 删除有序数组中的重复项Ⅰ a.思路 b.图解 c.代码 d.思考 删除有序数组中的重复项Ⅱ a.思路 b.图解 c.代码 d.思考 删除有序数组中的重复项Ⅰ a.思路 定义变量 int dest=0,cur=1,nums[cur]与nums[dest]逐一比较. nums[cur]!=nums[dest],将nums[cur]放入dest下一个位置,更新dest. nums[cur]!=nums[dest],cur移动. cur==numsSize,结束.返回dest+1. b.图解 c.

  • C语言详解float类型在内存中的存储方式

    目录 1.例子 2.浮点数存储规则 1.例子 int main() { int n = 9; float *pFloat = (float *)&n; printf("n的值为:%d\n",n); printf("*pFloat的值为:%f\n",*pFloat); *pFloat = 9.0; printf("num的值为:%d\n",n); printf("*pFloat的值为:%f\n",*pFloat); re

随机推荐