Linux進(jìn)程幾種退出方式的比較
導語(yǔ):當一個(gè)進(jìn)程結束了運行或在半途中終止了運行,那么內核就需要釋放該進(jìn)程所占用的系統資源。這包括進(jìn)程運行時(shí)打開(kāi)的文件,申請的內存等。
本文引用地址:http://dyxdggzs.com/article/201810/392683.htm進(jìn)程退出
Linux 下進(jìn)程的退出分為正常退出和異常退出兩種:
1.正常退出
a. 在main()函數中執行return
b.調用exit()函數
c.調用_exit()函數
2.異常退出
a.調用about函數
b.進(jìn)程收到某個(gè)信號,而該信號使程序終止
不管是哪種退出方式,系統最終都會(huì )執行內核中的同一代碼。這段代碼用來(lái)關(guān)閉進(jìn)程所用已打開(kāi)的文件描述符,釋放它所占用的內存和其他資源。
幾種退出方式的比較
1.exit和return 的區別
exit是一個(gè)函數,有參數。exit執行完后把控制權交給系統。
return是函數執行完后的返回,renturn執行完后把控制權交給調用函數。
2.exit和abort的區別
exit是正常終止進(jìn)程。
about是異常終止。
exit()和_exit()函數
exit和_exit函數都是用來(lái)終止進(jìn)程的。當程序執行到exit或_exit時(shí),系統無(wú)條件的停止剩下所有操作,清除各種數據結構,并終止本進(jìn)程的運行。
exit在頭文件stdlib.h中聲明,而_exit()聲明在頭文件unistd.h中聲明。 exit中的參數exit_code為0代表進(jìn)程正常終止,若為其他值表示程序執行過(guò)程中有錯誤發(fā)生。
exit()和_exit()的區別
_exit()執行后立即返回給內核,而exit()要先執行一些清除操作,然后將控制權交給內核。
調用_exit函數時(shí),其會(huì )關(guān)閉進(jìn)程所有的文件描述符,清理內存以及其他一些內核清理函數,但不會(huì )刷新流(stdin, stdout, stderr ...). exit函數是在_exit函數之上的一個(gè)封裝,其會(huì )調用_exit,并在調用之前先刷新流。
exit()函數與_exit()函數最大區別就在于exit()函數在調用exit系統之前要檢查文件的打開(kāi)情況,把文件緩沖區的內容寫(xiě)回文件。由于Linux的標準函數庫中,有一種被稱(chēng)作“緩沖I/O”的操作,其特征就是對應每一個(gè)打開(kāi)的文件,在內存中都有一片緩沖區。
每次讀文件時(shí),會(huì )連續的讀出若干條記錄,這樣在下次讀文件時(shí)就可以直接從內存的緩沖區讀取;同樣,每次寫(xiě)文件的時(shí)候也僅僅是寫(xiě)入內存的緩沖區,等滿(mǎn)足了一定的條件(如達到了一定數量或遇到特定字符等),再將緩沖區中的內容一次性寫(xiě)入文件。
這種技術(shù)大大增加了文件讀寫(xiě)的速度,但也給編程代來(lái)了一點(diǎn)兒麻煩。比如有一些數據,認為已經(jīng)寫(xiě)入了文件,實(shí)際上因為沒(méi)有滿(mǎn)足特定的條件,它們還只是保存在緩沖區內,這時(shí)用_exit()函數直接將進(jìn)程關(guān)閉,緩沖區的數據就會(huì )丟失。因此,要想保證數據的完整性,就一定要使用exit()函數。
通過(guò)一個(gè)函數實(shí)例來(lái)看看它們之間的區別:
函數實(shí)例1: exit.c
#include
#include
int main()
{
printf("using exit----n");
printf("This is the content in buffern");
exit(0);
}
執行結果為:
using exit----
This is the content in buffer
函數實(shí)例2:_exit.c
#include
#include
int main()
{
printf("using _exit--n");
printf("This is the content in buffer");
_exit(0);
}
執行結果為 :
using _exit--
printf函數就是使用緩沖I/O的方式,該函數在遇到“n”換行符時(shí)自動(dòng)的從緩沖區中將記錄讀出。所以exit()將緩沖區的數據寫(xiě)完后才退出,而_exit()函數直接退出。
大家也可以把函數實(shí)例2中的printf("This is the content in buffer");改為printf("This is the content in buffern")(即在printf中最后加一個(gè)n看運行結果是什么,為什么會(huì )產(chǎn)生這樣的結果呢?)
父子進(jìn)程終止的先后順序不同會(huì )產(chǎn)生不同的結果
1.父進(jìn)程先于子進(jìn)程終止
此種情況就是我們前面所用的孤兒進(jìn)程。當父進(jìn)程先退出時(shí),系統會(huì )讓init進(jìn)程接管子進(jìn)程 。
2.子進(jìn)程先于父進(jìn)程終止,而父進(jìn)程又沒(méi)有調用wait函數
此種情況子進(jìn)程進(jìn)入僵死狀態(tài),并且會(huì )一直保持下去直到系統重啟。子進(jìn)程處于僵死狀態(tài)時(shí),內核只保存進(jìn)程的一些必要信息以備父進(jìn)程所需。此時(shí)子進(jìn)程始終占有著(zhù)資源,同時(shí)也減少了系統可以創(chuàng )建的最大進(jìn)程數。
什么是 僵死狀態(tài)呢?一個(gè)已經(jīng)終止、但是其父進(jìn)程尚未對其進(jìn)行善后處理(獲取終止子進(jìn)程的有關(guān)信息,釋放它仍占有的資源)的進(jìn)程被稱(chēng)為僵死進(jìn)程(zombie)。
3.子進(jìn)程先于父進(jìn)程終止,而父進(jìn)程調用了wait函數
此時(shí)父進(jìn)程會(huì )等待子進(jìn)程結束。
評論