1.为什么会有atexit函数
很多时候我们需要在程序退出的时候做一些诸如释放资源的操作,但程序退出的方式有很多种,比如main()函数运行结束、在程序的某个地方用exit() 结束程序、用户通过Ctrl+C或Ctrl+break操作来终止程序等等,因此需要有一种与程序退出方式无关的方法来进行程序退出时的必要处理。方法就 是用atexit()函数来注册程序正常终止时要被调用的函数,也就是说它就是用来调用终止函数的。
下面我们在Linux环境下验证一下atexit函数调用 注册函数的顺序。
2.调用分析
atexit函数先注册4个fun函数然后等待2秒,再打印”main”,如果main函数中输出部分不加\n,则main函数要输出的内容会先放到标准输出缓冲区中,当main中调用exit函数的时候,会做一些自身清理功工作,同时刷新标准输出缓冲区中的内容。当执行到return 0 时,exit会自动调用这些已注册过的函数,但是由于压栈过程中先入后出的原则,所以先注册的函数最后执行。
3. atexit()函数的参数是一个函数指针,函数指针指向一个没有参数也没有返回值的函数。atexit()的函数原型是:int atexit (void (*)(void))
一个进程可以登记多达32个函数,这些函数将由exit自动调用,通常这32个函数被称为终止处理程序,并调用atexit函数来登记这些函数,atexit的参数是一个函数地址,当调用此函数时无需传递任何参数,该函数也不能返回值,atexit函数被称为终止处理程序注册程序,注册完成以后,当函数终止是exit()函数会主动的调用前面注册的各个函数,但是exit函数调用这些函数的顺序于这些函数登记的顺序是相反的,我认为这实质上是参数压栈造成的,参数由于压栈顺序而先入后出。同时如果一个函数被多次登记,那么该函数也将多次的执行。
4.exit函数
exit(int status)
_exit(int status)
Exit(int status)
这三个函数用于正常终止一个程序,_exit和_Exit立即进入内核,而exit则要做一些清理工作(调用执行个终止处理程序,关闭I/O流),再进入内核。三个函数所带的整形参数称为终止状态或退出状态。