Posted on 2009-04-14 15:28
Prayer 閱讀(979)
評論(0) 編輯 收藏 引用 所屬分類:
LINUX/UNIX/AIX
1. 僵死進程
當一個程序創建的子進程比父進程提前結束,內核仍然保存一些它的信息以便父進程會需要它 - 比如,父進程可能需要檢查子進程的退出狀態。為了得到這些
信息,父進程調用‘wait()’;當這個調用發生,內核可以丟棄這些信息。
在子進程終止后到父進程調用‘wait()’前的時間里,子進程被稱為‘僵死進程’ (‘zombie’)。(如果你用‘ps’,這個子進程會有一個‘Z’出現在它的
狀態區里指出這點。)
也就是說,當子進程調用 do_exit() 后,子進程就終止了,與次進程相關聯的所有資源都被釋放掉了(假設進程是這些資源的唯一使用者)。 進程不可
運行,并處于TASK_ZOMBIE狀態。它所占用的資源就是內核棧,thread_info結構和task_struct結構。此時進程存在的唯一目的就是向它的
父進程提供信息(子進程的退出代碼,也就是exit()提供的,存儲在task_struct中的exit_code成員)。父進程檢索到信息后(通過wait()系統調用),
或者通知內核那是無關的信息后(在某些系統上,你可以指令系統你對子進程的退出狀態沒有興趣,在SysV系統上,可以調用signal函數,
設置SIGCLD信號為 SIG_IGN,系統將不產生僵死進程, 詳細說明參見<<高級編程>>10.7節), 由進程所持有的剩余內存被釋放,歸還給系統使用。
2. 怎樣避免僵死進程的出現
需要卻認父進程為每個子進程的終止調用‘wait()’(或者‘waitpid()’, ‘wait3()’,等等); 或者,在某些系統上,你可以指令系統你對子進程的退出狀態沒有興趣。(譯者注:在SysV系統上,可以調用signal函數,設置SIGCLD信號為 SIG_IGN,系統將不產生僵死進程, 詳細說明參見<<高級編程>>10.7節)。
另一種方法是兩次‘fork()’,而且使緊跟的子進程直接退出,這樣造成孫子進程變成孤兒進程(orphaned),從而init進程將負責清除它。欲獲得做這個的程序,參看范例章節的函數‘fork2()’。
為了忽略子進程狀態,你需要做下面的步驟(查詢你的系統手冊頁以知道這是否正常工作):
struct sigaction sa;
sa.sa_handler = SIG_IGN;
#ifdef SA_NOCLDWAIT
sa.sa_flags = SA_NOCLDWAIT;
#else
sa.sa_flags = 0;
#endif
sigemptyset(&sa.sa_mask);
sigaction(SIGCHLD, &sa, NULL);
如果這是成功的,那么‘wait()’函數集將不再正常工作;如果它們中任何一個被調用,它們將等待直到*所有*子進程已經退出,然后返回失敗,并且 ‘errno==ECHILD’。
另一個技巧是捕獲SIGCHLD信號,然后使信號處理程序調用‘waitpid()’或 ‘wait3()’。參見范例章節的完整程序。
參考文獻:
http://www.juntuan.net/hkbc/nixbc/unix/x178.htm
《APUE》
《LKD》
文章出處:http://www.diybl.com/course/6_system/linux/Linuxjs/200888/135107.html