当内核执行C程序时(使用一个exec函数,即创建),在调用main前先调用一个特殊的启动例程。可执行程序文件将此启动例程指定为程序的起始地址——这是由C编译器调用设
置。启动例程从内核取得命令行参数和环境变量值,然后为按上述方式调用main函数做好安排。
当终止一个进程时,有8种方式使进程终止,其中5种为正常终止,它们是:
- 从main函数返回;
- 调用exit;
- 调用_exit或_Exit;
- 最后一个线程从启动例程返回;
- 从最后一个线程调用pthread_exit;
异常终止方式有3种,它们是:
- 调用abort;
- 接到一个信号;
- 最后一个线程对取消请求做出响应;
从函数返回后会立即调用exit函数;
如果将启动例程以C代码形式表示(实际上该例程常常用汇编语言编写),则它调用main函数的形式可能是:
启动例程
{
.....
exit(main(argc,argv));
.....
}
_exit和_Exit立即进入内核,exit则先执行一些清理处理,然后返回内核。C程序的启动分析如下:
在函数终止时,我们可以调用atexit函数来登记终止处理程序(在进程终止时由exit自动调用)。
atexit函数的使用说明如下:
注:exit调用这些函数的顺序与它们登记时候的顺序相反,同一函数如若登记多次,也会被多次调用;
示例如下:exitHandler.c
/*********************************************************************
*File : ExitHandler.c
*Author : fzz email:fzz_joy@163.com
*Versions : 2015/8/16 V1.0
*Description : it calls when process is ending.
**********************************************************************/
#include<stdio.h>
#include<stdlib.h>
static void my_exit1(void);
static void my_exit2(void);
int main()
{
if((atexit(my_exit2)!=0))
printf("can't register my_exit2!");
if((atexit(my_exit1)!=0))
printf("can't register my_exit1!");
if((atexit(my_exit1)!=0))
printf("can't register my_exit1!");
printf("main is done.\n");
return 0;
}
static void my_exit1(void)
{
printf("first exit handler!\n");
}
static void my_exit2(void)
{
printf("second exit handler!\n");
}
运行结果如下:
参考书籍:《Unix环境高级编程(第三版)》P158