一、atexit函数
atexit函数是一个特殊的函数,它是在正常程序退出时调用的函数,我们把他叫为登记函数(函数原型:int atexit (void (*)(void)))
一个进程可以登记32个函数,这些函数由exit自动调用,这些函数被称为终止处理函数,atexit函数可以登记这些函数。exit调用终止处理函数的顺序和atexit登记的顺序相反,如果一个函数被多次登记,也会被多次调用。
二、 进程终止
以下函数在调用时程序异常或者正常终止:
进程终止的方式有8种,前5种为正常终止,后三种为异常终止:
1、 从main函数返回;
2、 调用exit函数;
3 、调用_exit或_Exit;
4 、最后一个线程从启动例程返回;
5、 最后一个线程调用pthread_exit;
6 、调用abort函数;
7 、接到一个信号并终止;
8、 最后一个线程对取消请求做出响应。
下面我们来验证atexit的调用顺序和退出顺序:
代码如下:
#include<stdio.h>
#include<stdlib.h>
void func1()
{
printf("hello func1\n");
}
void func2()
{
printf("hello func2\n");
}
void func3()
{
printf("hello func3\n");
}
int main()
{
atexit(func1);
atexit(func2);
atexit(func3);
sleep(3);
printf("hello main\n");
exit(0);
}
在Linux 下如下所示:
运行结果:
我们可以看到atexit函数的调用顺序是和登记顺序相反的。
三、过程分析
atexit函数先注册三个func函数,然后等待3秒,再打印”hello main”(如果main函数中输出部分不加\n,则main函数要输出的内容会先放到标准输出缓冲区中,当main中调用exit函数的时候,会做一些自身清理工作,同时刷新标准输出缓冲区中的内容),当执行到exit(0)时,exit会自动调用这些已注册过的函数,但是由于压栈过程中先进后出的原则,所以先注册的函数最后执行。
关于atexit:
一个进程可以登记多达32个函数,这些函数将由exit自动调用,通常这32个函数被称为终止处理程序,并调用atexit函数来登记这些函数,atexit的参数是一个函数地址,当调用此函数时无须传递任何参数,该函数也不能返回值,atexit函数称为终止处理程序注册程序,注册完成以后,当函数终止是exit()函数会主动的调用前面注册的各个函数,但是exit函数调用这些函数的顺序于这些函数登记的顺序是相反的,这实质上应该是参数压栈造成的,参数由于压栈顺序是先进后出。同时如果一个函数被多次登记,那么该函数也将多次的执行。
关于exit:
exit函数运行时首先会执行由atexit()函数登记的函数,然后会做一些自身的清理工作,同时刷新所有输出流、关闭所有打开的流并且关闭通过标准I/O函数tmpfile()创建的临时文件。
exit()函数用于在程序运行的过程中随时结束程序,exit的参数state是返回给操作系统,返回0表示程序正常结束,非0表示程序非正常结束。