淺析結束程序函數exit, _exit,atexit的區(qū)別
多時候我們需要在程序退出的時候做一些諸如釋放資源的操作,但程序退出的方式有很多種,比如main()函數運行結束、在程序的某個地方用exit() 結束程序、用戶通過Ctrl+C或Ctrl+break操作來終止程序等等,因此需要有一種與程序退出方式無關的方法來進行程序退出時的必要處理。方法就 是用atexit()函數來注冊程序正常終止時要被調用的函數。
atexit()函數的參數是一個函數指針,函數指針指向一個沒有參數也沒有返回值的函數。atexit()的函數原型是:int atexit (void (*)(void));
在一個程序中最多可以用atexit()注冊32個處理函數,這些處理函數的調用順序與其注冊的順序相反,也即最先注冊的最后調用,最后注冊的最先調用。
#include <stdlib.h>
#include <stdio.h>
void fun()
{
printf("fun/n");
}
int main()
{
atexit(fun);
printf("hello/n");
return 0;
}
// int atexit(void (*func)()) // 見 <stdlib.h> 中定義
// {
// func();
// return 0;
// }
上面的代碼將輸出
hello
fun
而把紅色的注釋代碼去掉之后,由于Interpositioning行為,重定義了庫函數,使atexit僅僅表現(xiàn)為一個普通的函數
因此輸出
fun
hello
#include <stdlib.h>
#define EXIT_FAILURE ...
#define EXIT_SUCCESS ...
void exit (int status);
void _Exit(int status); //C99
void abort(void);
int atexit(void (*func)(void)));
exit、_Exit與abort函數使程序終止,控制并不返回到這些函數的調用者。
函數exit正常終止程序,并進行下列清理操作:
1.(進對標準C語言)所有想atexit函數注冊的函數按與注冊時相反的順序調用,注冊幾次就調用幾次。
2. 刷新打開的是輸出流,關閉所有打開的數據流。
3. 刪除tepfile函數生成的文件。
4. 控制返回宿主環(huán)境,提供狀態(tài)值。
按照許多系統(tǒng)中的習慣,status值為0表示終止程序成功,用非0值表示異常終止。標準C語言中數值0和宏 EXIT_SCCESS的值表示終止成功,宏EXIT_FAILURE的值表示終止不成功,其他值的含義由實現(xiàn)定義。從函數main返回一個整數值相當于用這個值調用exit函數。
函數_Exit與exit函數不同之處在于既不調用atexit注冊的退出處理器,也不調用singal注冊的信號處理器。是否進行其他清理操作由實現(xiàn)定義,如關閉所有打開的數據流。_Exit是C99增加的,傳統(tǒng)上有些實現(xiàn)用名為_exit的函數提供類似功能。
abort函數使程序異常終止,不調用向atexit注冊的函數。abort是否引起清理操作由實現(xiàn)定義,向宿主系統(tǒng)返回的狀態(tài)值也由實現(xiàn)定義,但應表示為“不成功”。在標準C語言和許多傳統(tǒng)實現(xiàn)中,調用abort轉換成可以捕獲的特殊信號(標準C語言中為 SIGABRT)。如果信號被忽略或處理器返回,則標準C語言實現(xiàn)仍然終止程序,而其他實現(xiàn)可能使abort函數返回調用者。斷言失敗也會調用 abort。
atexit函數是標準C語言中增加的,它注冊一個函數,使得調用exit時或函數main返回時會調用這個函數。程序異常終止時(如用abort或raise終止),不調用注冊的函數。實現(xiàn)應允許至少注冊32個函數。如果注冊成功,則atexit函數返回0,否則返回非0值,函數無法注銷。所有向atexit函數注冊的函數按與注冊相反的順序調用,然后再由atexit函數完成所有標準清理操作。每個函數不代參數調用,應具有返回類型void。注冊函數不能引用任何不是自己定義的存儲類為auto或 register的對象(例如通過指針引用)。函數注冊幾次就會在此時調用幾次。
指針函數使用示例:
#include<stdlib.h>
#include<stdio.h>
typedef void (*pFunc)(float a);
// int atexit(void (*func)())
// {
// func();
// return 0;
// }
int atexitf( void (*func)(float),float a)
{
func(a);
return 0;
}
void test(float a)
{
printf("test %f/n",a);
}
// void fun()
// {
// printf("fun/n");
// }
int main()
{
pFunc pFunc1 = (pFunc)test; // 函數指針賦值
pFunc1(4.5);
/* atexit(fun);*/
atexitf(pFunc1,3.66);
atexitf(test,3.66);
//atexitf(test(4.3));
printf("hello/n");
getchar();
return 0;
}
相關文章
OpenCV使用稀疏光流實現(xiàn)視頻對象跟蹤的方法詳解
這篇文章主要為大家詳細介紹了OpenCV如何使用稀疏光流實現(xiàn)視頻對象跟蹤功能,文中的示例代碼講解詳細,具有一定的借鑒價值,需要的可以參考一下2023-02-02C++實現(xiàn)LeetCode(82.移除有序鏈表中的重復項之二)
這篇文章主要介紹了C++實現(xiàn)LeetCode(82.移除有序鏈表中的重復項之二),本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下2021-07-07C語言字符串函數,字符函數,內存函數使用及模擬實現(xiàn)
這篇文章主要介紹了C語言字符串函數,字符函數,內存函數使用及模擬實現(xiàn),文章圍繞主題展開詳細的內容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-09-09