一個進程在調(diào)用exit命令結(jié)束自己的生命的時候,其實它并沒有真正的被銷毀,而是留下一個稱為僵死進程(Zombie)的數(shù)據(jù)結(jié)構(gòu)(系統(tǒng)調(diào)用exit,它的作用是使進程退出,但也僅僅限于將一個正常的進程變成一個僵死進程,并不能將其完全銷毀)。
在每個進程退出的時候,內(nèi)核釋放該進程所有的資源,包括打開的文件,占用的內(nèi)存等,但是仍然為其保留一定的信息 (包括進程號the process ID,退出狀態(tài)the term ination status of the process,運行時間the amount of CPU time taken by the process等),直到父進程通過wait/waitpid來取時才釋放。此時該進程處于僵死狀態(tài),該進程成為僵死進程 (Zombie Process)。 這保證了父進程可以獲取到子進程結(jié)束時的狀態(tài)信息。
在Linux進程的狀態(tài)中,僵死進程是非常特殊的一種,它已經(jīng)放棄了幾乎所有內(nèi)存空間,沒有任何可執(zhí)行代碼,也不能被調(diào)度,僅僅在進程列表中保留一個位置,記載該進程的退出狀態(tài)等信息供其他進程收集,除此之外,僵死進程不再占有任何內(nèi)存空間。它需要它的父進程來為它收尸,如果他的父進程沒安裝SIGCHLD信號處理函數(shù)調(diào)用wait或w aitpid()等待子進程結(jié)束,又沒有顯式忽略該信號,那么它就一直保持僵死狀態(tài),如果這時父進程結(jié)束了,僵死的子進程成為"孤兒進程 ",過繼給 1號進程 init,init始終會負(fù)責(zé)清理僵死進程,它產(chǎn)生的所有僵死進程也跟著消失(每個進程結(jié)束的時候,系統(tǒng)都會掃描當(dāng)前系統(tǒng)中所運行的所有進程,看有沒有哪個進程是剛剛結(jié)束的這個進程的子進程,如果是的話,就由Init來接管他,成為他的父進程)。但是如果如果父進程是一個循環(huán),不會結(jié)束,那么子進程就會一直保持僵死狀態(tài),這就是為什么系統(tǒng)中有時會有很多的僵死進程。怎么查看僵死進程,利用命令ps,可以看到有標(biāo)記為Z的進程就是僵死進程。
如果父進程不調(diào)用wait/w aitpid的話,那么保留的那段信息就不會釋放,其進程號會一定被占用,但是系統(tǒng)所能使用的進程號是有限的,如果產(chǎn)生了大量的僵死進程,將因為沒有可用的進程號而導(dǎo)致系統(tǒng)不能產(chǎn)生新的進程。
1、父進程通過 wait和 w aitpid等函數(shù)等待子進程結(jié)束,這會導(dǎo)致父進程掛起。
2、如果父進程很忙,那么可以用signal函數(shù)為SIGCHLD安裝信號處理函數(shù)。子進程結(jié)束后,父進程會收到該信號,可以在信號處理函數(shù)中調(diào)用wait回收 。
3、如果父進程不關(guān)心子進程什么時候結(jié)束,那么可以用signal(SIGCHLD,SIG_IGN)通知內(nèi)核,自己對子進程的結(jié)束不感興趣,那么子進程結(jié)束后,內(nèi)核會回收,并不再給父進程 發(fā)送信號。
或用sigaction函數(shù)為SIGCHLD設(shè)置SA_NOCLDWAIT,這樣子進程結(jié)束后,就不會進入僵死狀態(tài)
4、fork兩次,父進程 fork一個子進程,然后繼續(xù)工作,子進程fork一個孫進程后退出,那么孫進程被init接管,孫進程結(jié)束后,init會回收。不過子進程的回收還要父進程來做。
waitpid(pid,&nStatus,0);//等待子進程結(jié)束,否則子進程會成為僵死進程,一直存在,即便子進程已結(jié)束執(zhí)行
exit(0);//子進程退出,孫進程過繼給init進程,其退出狀態(tài)也由init進程處理,與原有父進程無關(guān)