思维导图
学习内容
进程终止是进程控制里面的一个重要的知识,通过这一篇博客,我们可以学习到进程终止的概念,进程终止的三种情况,进程终止的退出码和退出信号,最后在来学习进程是如何进行终止的。
学习目标
- 进程终止的概念
- echo内建命名
- 进程退出码
- 进程终止的三种情况
- 进程是如何进行终止
一、进程终止的概念
在计算机系统中,进程是操作系统分配资源的基本单位,而进程终止则是指操作系统因为某种原因结束一个进程的执行。这可能是因为进程完成了某种任务,也可能是因为出现了错误或异常。
创建一个进程的过程是:先创建一个内核的部分数据结构,再进行复制代码和数据。那么终止一个进程的过程是和创建一个进程的顺序是相反的,终止一个进程的本质是:释放曾经的代码和数据所占据的空间,释放内核数据结构。
二、echo 内建命令
echo内建命令的概念:打印的是bash内部的变量数据。
?符号的意义:父进程bash获取到最近一个子进程退出的退出码,0表示正常退出,非0表示异常退出。这个返回值需要让父进程得到,告诉父进程,子进程将任务完成的怎么样。
不是**?符号**可以将最近一个子进程退出的退出码获取吗?但是上图中为什么获取了两个不同的子进程退出的退出码。
因为echo命令也是一个进程,当使用完echo命令后,变成了最近一次子进程,又因为echo命令正常退出,所以最近一次子进程退出的退出码是0。
三、进程退出码
3.1 strerror函数
3.1.1 strerror函数的用途获取指向错误信息字符串的指针。
3.1.2 strerror函数的介绍
cpp
#include <string.h>
char* strerror(int errnum)
// 返回值为char * 类型,指向描述错误错误的错误字符串的指针
3.1.3 利用strerror函数来获取C语言中所有的错误信息
cpp
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main()
{
int errcode = 0;
for(errcode = 0; errcode <= 255; errcode++)
{
printf("%d %s\n", errcode, strerror(errcode));
}
return 100;
}
3.2 父进程为什么要获取子进程的退出码?
因为父进程要知道子进程完成任务的情况(成功或者失败,失败的原因是什么),要对子进程进行负责。
3.3 自定义退出码
我们可以通过联合体为自定义退出码进行赋值,将退出码置为全局变量,在每一个函数的每一种结果中都需要将退出码进行修改,在通过接收退出码的值将退出码的值翻译为错误信息的字符串。
cpp
#include <stdio.h>
enum{
Success = 0,
Div_Zero,
Mod_Zero,
};
int errcode = Success;
int Div(int a, int b)
{
if(b == 0)
{
errcode = Div_Zero;
return -1;
}
else{
errcode = Success;
return a / b;
}
}
char* geterror(int errno)
{
switch(errno)
{
case Success:
return "Success";
case Div_Zero:
return "Div_Zero";
case Mod_Zero:
return "Mod_Zero";
dafault:
return NULL;
}
}
int main()
{
int tmp1 = Div(10, 1);
printf("answer: %d Error: %s\n", tmp1, geterror(errcode));
int tmp2 = Div(10, 0);
printf("answer: %d Error: %s\n", tmp2, geterror(errcode));
return 0;
}
四、进程终止的三种情况
4.1 通过进程的退出码决定的进程终止的情况
- 代码跑完,结果正确
- 代码跑完,结果不正确
4.2 通过进程的退出信号决定的进程终止的情况
最后一种情况:代码执行时,出现了异常,提前退出了,退出码没有意义。
- 编译运行时,崩溃了,操作系统发现了你的进程做了不该做的事情,操作系统杀死了进程。
- 一旦进程出现异常,退出码没有意义
进程异常的退出信号:
我们可以使用kill命令来显示进程异常的退出信号所代表的含义:
cpp
kill -l
现在我们可以通过做个实验,来查看进程异常退出是否需要错误信号。
cpp
#include <stdio.h>
#include <unistd.h>
int main()
{
while (1)
{
printf("I am process, pid = %d\n", getpid());
sleep(1);
}
return 0;
}
当代码中出现一些异常错误时,进程也会发出错误信号,杀死进程。
总结: 判断一个进程的退出情况,我只需要检查进程是否出现异常;如果没有出现异常,就检查进程的退出码。衡量一个进程退出,我们只需要两个数字:退出码和退出信号。
五、进程如何终止
- main函数调用return表示进程终止(普通函数的return,是退出函数)
- 代码调用exit函数,在进程退出后,会冲刷缓冲区
- 系统调用_exit函数,在进程退出后,不会冲刷缓冲区
我们可以通过代码来观察现象:
cpp
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
printf("Hello world");
exit(1);
return 0;
}
cpp
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main()
{
printf("Hello world");
_exit(1);
return 0;
}