进程的终止
-
_exit()
通常进程有两种终止方式,一种是异常终止,当进程接受到某一个信号,而该信号的默认动作又是终止当前进程(term | core)时,进程就会被异常终止
此外,进程可以使用_exit()系统调用来正常终止
#include<unistd.h> void _exit(int status); //return 0 表示正常退出,非0表示异常退出
参数status定义了进程的终止状态,父进程可以使用
wait
来获取该状态status虽然为int类型,但是只有低8位会被父进程使用,还可以返回常量:EXIT_SUCCESS,EXIT_FAILURE来表示返回的状态
虽然status的取值范围为0~255(2^8 - 1),但是最好不要使用128以上的返回值,因为当使用信号来终止进程时,shell会将变量$?(上一个命令的返回结果,0表示没有错误)置为128 + 信号值,可能会与返回值相等,造成冲突
-
exit()
exit是标准C语言函数库,其对_exit()进行了封装
#include<stdlib.h> void exit(int status) //return 0表示正常退出,非0表示异常退出
使用exit时会执行的动作如下:
- 调用退出处理程序(看下文)
- 刷新stdio缓冲区
- 执行_exit(status)
顺便介绍一下return 关键字,当程序使用return关键字或者在void函数中执行到程序末尾时,会隐式地执行exit(),
return n ------->exit(n), 没有指定返回值的话的话就是exit(0)(也不一定,C99这样规定,但是C89未做定义)
-
退出处理程序
退出处理程序是一个由程序设计者提供的函数,可与进程声明周期的任意时点注册,并且会在进程调用exit()时自动调用
如果程序直接调用_exit()或者是因为信号而导致异常终止,那么信号处理程序便不会执行
可以采用建立信号处理程序,在信号处理程序中设置标志位,进而在主程序中使用退出处理程序,来弥补一些限制
注册退出处理程序
- atexit()
#include<stdlib.h> int atexit(void (*func)(void)) //return 0 成功,非0失败
atexit会将func添加到一个函数列表,进程终止时会调用该函数列表中的所有函数、
当注册多个退出处理程序时,这些函数的执行顺醋与注册顺序相反
在调用某一个退出处理程序时,如果调用发生异常导致该函数无法返回,那么剩余的所有处 理函数均不会执行
-
通过fork调用创建的子进程会继承附近产注册的退出处理函数(复制文本段)
通过exec调用会移除所有已经注册的退出处理程序(初始化所有段)
-
on_exit()
这是一个glibc非标准函数,但是功能更加强大
#define _DEFAULT_SOURCE #include<stdlib.h> int on_exit(void(*func)(int, void*), void* arg) //return 0 成功,非0失败
func函数中的int参数为调用exit(status)时给定的status参数,void*类型参数为注册时(即使用on_exit(func, arg)),用户自己给定的参数arg,用于在退出处理程序实现一些自定义功能
-
fork(), stdio缓冲区,以及_exit()之间的交互
由于stdio缓冲区是在进程的用户空间进行维护的,所以使用fork时就会理所当然的将其拷贝到子进程的用户空间,在父子进程调用exit()会刷新各自的stdio缓冲区,由此可能就会导致重复的输出结果
如何避免?
- 可以在调用fork()之前使用函数fflush()来手动刷新缓冲区,当然也可以使用setvbuf或者setbuf来关闭缓冲功能
- 子进程使用_exit()终止而不是exit(),这样就不会刷新exit()缓冲区