結束程序函數exit、 _exit、 atexit區別

多時候我們需要在程序退出的時候做一些諸如釋放資源的操作,但程序退出的方式有很多種,比如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僅僅表現爲一個普通的函數

因此輸出

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. 控制返回宿主環境,提供狀態值。

按照許多系統中的習慣,status值爲0表示終止程序成功,用非0值表示異常終止。標準C語言中數值0和宏 EXIT_SCCESS的值表示終止成功,宏EXIT_FAILURE的值表示終止不成功,其他值的含義由實現定義。從函數main返回一個整數值相當於用這個值調用exit函數。

函數_Exit與exit函數不同之處在於既不調用atexit註冊的退出處理器,也不調用singal註冊的信號處理器。是否進行其他清理操作由實現定義,如關閉所有打開的數據流。_Exit是C99增加的,傳統上有些實現用名爲_exit的函數提供類似功能。

abort函數使程序異常終止,不調用向atexit註冊的函數。abort是否引起清理操作由實現定義,向宿主系統返回的狀態值也由實現定義,但應表示爲“不成功”。在標準C語言和許多傳統實現中,調用abort轉換成可以捕獲的特殊信號(標準C語言中爲 SIGABRT)。如果信號被忽略或處理器返回,則標準C語言實現仍然終止程序,而其他實現可能使abort函數返回調用者。斷言失敗也會調用 abort。

atexit函數是標準C語言中增加的,它註冊一個函數,使得調用exit時或函數main返回時會調用這個函數。程序異常終止時(如用abort或raise終止),不調用註冊的函數。實現應允許至少註冊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;
}

發佈了78 篇原創文章 · 獲贊 54 · 訪問量 35萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章