佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > Linux系統(tǒng)命令中exit與exit的區(qū)別

Linux系統(tǒng)命令中exit與exit的區(qū)別

熱門標(biāo)簽:電話機(jī)器人搭建 服務(wù)器配置 硅谷的囚徒呼叫中心 美團(tuán) 百度競價點擊價格的計算公式 解決方案 外呼系統(tǒng) 家政服務(wù)網(wǎng)絡(luò)

注:exit()就是退出,傳入的參數(shù)是程序退出時的狀態(tài)碼,0表示正常退出,其他表示非正常退出,一般都用-1或者1,標(biāo)準(zhǔn)C里有EXIT_SUCCESSEXIT_FAILURE兩個宏,用exit(EXIT_SUCCESS);可讀性比較好一點。

作為系統(tǒng)調(diào)用而言,_exitexit是一對孿生兄弟,它們究竟相似到什么程度,我們可以從Linux的源碼中找到答案:

#define __NR__exit __NR_exit /* 摘自文件include/asm-i386/unistd.h第334行 */

"__NR_"是在Linux的源碼中為每個系統(tǒng)調(diào)用加上的前綴,請注意第一個exit前有2條下劃線,第二個exit前只有1條下劃線。 這時隨便一個懂得C語言并且頭腦清醒的人都會說,_exit和exit沒有任何區(qū)別,但我們還要講一下這兩者之間的區(qū)別,這種區(qū)別主要體現(xiàn)在它們在函數(shù)庫中的定義。_exit在Linux函數(shù)庫中的原型是:

#i ncludeunistd.h> void _exit(int status);

和exit比較一下,exit()函數(shù)定義在stdlib.h中,而_exit()定義在unistd.h中,從名字上看,stdlib.h似乎比 unistd.h高級一點,那么,它們之間到底有什么區(qū)別呢? _exit()函數(shù)的作用最為簡單:直接使進(jìn)程停止運(yùn)行,清除其使用的內(nèi)存空間,并銷毀其在內(nèi)核中的各種數(shù)據(jù)結(jié)構(gòu);exit() 函數(shù)則在這些基礎(chǔ)上作了一些包裝,在執(zhí)行退出之前加了若干道工序,也是因為這個原因,有些人認(rèn)為exit已經(jīng)不能算是純粹的系統(tǒng)調(diào)用。 exit()函數(shù)與_exit()函數(shù)最大的區(qū)別就在于exit()函數(shù)在調(diào)用exit系統(tǒng)調(diào)用之前要檢查文件的打開情況,把文件緩沖區(qū)中的內(nèi)容寫回文件,就是"清理I/O緩沖"。 

exit()在結(jié)束調(diào)用它的進(jìn)程之前,要進(jìn)行如下步驟:

1.調(diào)用atexit()注冊的函數(shù)(出口函數(shù));按ATEXIT注冊時相反的順序調(diào)用所有由它注冊的函數(shù),這使得我們可以指定在程序終止時執(zhí)行自己的清理動作.例如,保存程序狀態(tài)信息于某個文件,解開對共享數(shù)據(jù)庫上的鎖等.

2.cleanup();關(guān)閉所有打開的流,這將導(dǎo)致寫所有被緩沖的輸出,刪除用TMPFILE函數(shù)建立的所有臨時文件.

3.最后調(diào)用_exit()函數(shù)終止進(jìn)程。

_exit3件事(man): 1Any open file descriptors belonging to the process are closed 2,any children of the process are inherited by process 1, init 3the process‘s parent is sent a SIGCHLD signal

exit執(zhí)行完清理工作后就調(diào)用_exit來終止進(jìn)程。

此外,另外一種解釋:

簡單的說,exit函數(shù)將終止調(diào)用進(jìn)程。在退出程序之前,所有文件關(guān)閉,緩沖輸出內(nèi)容將刷新定義,并調(diào)用所有已刷新的“出口函數(shù)”(由atexit定義)。

_exit:該函數(shù)是由Posix定義的,不會運(yùn)行exit handler和signal handler,在UNIX系統(tǒng)中不會flush標(biāo)準(zhǔn)I/O流。

簡單的說,_exit終止調(diào)用進(jìn)程,但不關(guān)閉文件,不清除輸出緩存,也不調(diào)用出口函數(shù)。

共同:

不管進(jìn)程是如何終止的,內(nèi)核都會關(guān)閉進(jìn)程打開的所有file descriptors,釋放進(jìn)程使用的memory!

更詳細(xì)的介紹:

Calling exit() The exit() function causes normal program termination.

The exit() function performs the following functions:

 1. All functions registered by the Standard C atexit() function are called in the reverse order of registration. If any of these functions calls exit(), the results are not portable. 2. All open output streams are flushed (data written out) and the streams are closed.

3. All files created by tmpfile() are deleted.

4. The _exit() function is called. Calling _exit() The _exit() function performs operating system-specific program termination functions. These include: 1. All open file descriptors and directory streams are closed.

 2. If the parent process is executing a wait() or waitpid(), the parent wakes up and status is made available.

3. If the parent is not executing a wait() or waitpid(), the status is saved for return to the parent on a subsequent wait() or waitpid(). 4. Children of the terminated process are assigned a new parent process ID. Note: the termination of a parent does not directly terminate its children. 5. If the implementation supports the SIGCHLD signal, a SIGCHLD is sent to the parent. 6. Several job control signals are sent.

為何在一個fork的子進(jìn)程分支中使用_exit函數(shù)而不使用exit函數(shù)? ‘exit()’‘_exit()’有不少區(qū)別在使用‘fork()’,特別是‘vfork()’時變得很 突出。

‘exit()’‘_exit()’的基本區(qū)別在于前一個調(diào)用實施與調(diào)用庫里用戶狀態(tài)結(jié)構(gòu)(user-mode constructs)有關(guān)的清除工作(clean-up),而且調(diào)用用戶自定義的清除程序 (自定義清除程序由atexit函數(shù)定義,可定義多次,并以倒序執(zhí)行),相對應(yīng),_exit函數(shù)只為進(jìn)程實施內(nèi)核清除工作。 在由‘fork()’創(chuàng)建的子進(jìn)程分支里,正常情況下使用‘exit()’是不正確的,這是 因為使用它會導(dǎo)致標(biāo)準(zhǔn)輸入輸出(stdio: Standard Input Output)緩沖區(qū)被清空兩次,而且臨時文件被出乎意料的刪除(臨時文件由tmpfile函數(shù)創(chuàng)建在系統(tǒng)臨時目錄下,文件名由系統(tǒng)隨機(jī)生成)。在C++程序中情況會更糟,因為靜態(tài)目標(biāo)(static objects)的析構(gòu)函數(shù)(destructors)可以被錯誤地執(zhí)行。(還有一些特殊情況,比如守護(hù)程序,它們的父進(jìn)程需要調(diào)用‘_exit()’而不是子進(jìn)程;適用于絕大多數(shù)情況的基本規(guī)則是,‘exit()’在每一次進(jìn)入main’函數(shù)后只調(diào)用一次。) 在由‘vfork()’創(chuàng)建的子進(jìn)程分支里,‘exit()’的使用將更加危險,因為它將影響父進(jìn)程的狀態(tài)。

#include sys/types.h>; #include stdio.h> int glob = 6; /* external variable in initialized data */ int main(void) { int var; /* automatic variable on the stack */ pid_t pid; var = 88; printf("before vfork\n"; /* we don‘t flush stdio */ if ( (pid = vfork())  0) printf("vfork error\n"; else if (pid == 0) { /* child */ glob++; /* modify parent‘s variables */ var++; exit(0); /* child terminates */ //子進(jìn)程中最好還是用_exit(0)比較安全。 } /* parent */ printf("pid = %d, glob = %d, var = %d\n", getpid(), glob, var); exit(0); } 在Linux系統(tǒng)上運(yùn)行,父進(jìn)程printf的內(nèi)容輸出:pid = 29650, glob = 7, var = 89

子進(jìn)程 關(guān)閉的是自己的, 雖然他們共享標(biāo)準(zhǔn)輸入、標(biāo)準(zhǔn)輸出、標(biāo)準(zhǔn)出錯等 打開的文件, 子進(jìn)程exit時,也不過是遞減一個引用計數(shù),不可能關(guān)閉父進(jìn)程的,所以父進(jìn)程還是有輸出的。

但在其它UNIX系統(tǒng)上,父進(jìn)程可能沒有輸出,原 因是子進(jìn)程調(diào)用了e x i t,它刷新關(guān)閉了所有標(biāo)準(zhǔn)I / O流,這包括標(biāo)準(zhǔn)輸出。雖然這是由子進(jìn)程執(zhí)行的,但卻是在父進(jìn)程的地址空間中進(jìn)行的,所以所有受到影響的標(biāo)準(zhǔn)I/O FILE對象都是在父進(jìn)程中的。當(dāng)父進(jìn)程調(diào)用p r i n t f時,標(biāo)準(zhǔn)輸出已被關(guān)閉了,于是p r i n t f返回- 1。

Linux的標(biāo)準(zhǔn)函數(shù)庫中,有一套稱作"高級I/O"的函數(shù),我們熟知的printf()、fopen()、fread()、fwrite()都在此 列,它們也被稱作"緩沖I/Obuffered I/O",其特征是對應(yīng)每一個打開的文件,在內(nèi)存中都有一片緩沖區(qū),每次讀文件時,會多讀出若干條記錄,這樣下次讀文件時就可以直接從內(nèi)存的緩沖區(qū)中讀取,每次寫文件的時候,也僅僅是寫入內(nèi)存中的緩沖區(qū),等滿足了一定的條件(達(dá)到一定數(shù)量,或遇到特定字符,如換行符和文件結(jié)束符EOF), 再將緩沖區(qū)中的 內(nèi)容一次性寫入文件,這樣就大大增加了文件讀寫的速度,但也為我們編程帶來了一點點麻煩。如果有一些數(shù)據(jù),我們認(rèn)為已經(jīng)寫入了文件,實際上因為沒有滿足特 定的條件,它們還只是保存在緩沖區(qū)內(nèi),這時我們用_exit()函數(shù)直接將進(jìn)程關(guān)閉,緩沖區(qū)中的數(shù)據(jù)就會丟失,反之,如果想保證數(shù)據(jù)的完整性,就一定要使用exit()函數(shù)。

Exit的函數(shù)聲明在stdlib.h頭文件中。

_exit的函數(shù)聲明在unistd.h頭文件當(dāng)中。

下面的實例比較了這兩個函數(shù)的區(qū)別。printf函數(shù)就是使用緩沖I/O的方式,該函數(shù)在遇到“\n”換行符時自動的從緩沖區(qū)中將記錄讀出。實例就是利用這個性質(zhì)進(jìn)行比較的。

exit.c源碼

#include stdlib.h> #include stdio.h> int main(void) { printf("Using exit...\n"); printf("This is the content in buffer"); exit(0); }

輸出信息:

Using exit...

This is the content in buffer

#include unistd.h> #include stdio.h> int main(void) { printf("Using exit...\n"); //如果此處不加“\n”的話,這條信息有可能也不會顯示在終端上。 printf("This is the content in buffer"); _exit(0); }

則只輸出:

Using exit...

說明:在一個進(jìn)程調(diào)用了exit之后,該進(jìn)程并不會馬上完全消失,而是留下一個稱為僵尸進(jìn)程(Zombie)的數(shù)據(jù)結(jié)構(gòu)。僵尸進(jìn)程是一種非常特殊的進(jìn)程,它幾乎已經(jīng)放棄了所有的內(nèi)存空間,沒有任何可執(zhí)行代碼,也不能被調(diào)度,僅僅在進(jìn)程列表中保留一個位置,記載該進(jìn)程的退出狀態(tài)等信息供其它進(jìn)程收集,除此之外,僵尸進(jìn)程不再占有任何內(nèi)存空間。

#include stdio.h>;

int main() { printf("%c", ‘c‘); _exit(0); }

標(biāo)簽:臨沂 撫州 韶關(guān) 北海 烏蘭察布 邢臺 南昌 防城港

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Linux系統(tǒng)命令中exit與exit的區(qū)別》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    罗城| 建始县| 龙胜| 衢州市| 金乡县| 饶平县| 名山县| 昌图县| 长阳| 洛浦县| 郴州市| 米泉市| 商丘市| 西和县| 含山县| 深州市| 凤城市| 瑞金市| 山阳县| 扬州市| 德昌县| 温泉县| 松江区| 平阳县| 溧水县| 阜城县| 松潘县| 桐梓县| 阿坝县| 崇仁县| 东光县| 满城县| 改则县| 龙海市| 那曲县| 娱乐| 巨野县| 墨玉县| 当涂县| 仙居县| 贵定县|