詳解linux下避免僵尸進(jìn)程的幾種方法
linux下我們可以調(diào)用fork函數(shù)創(chuàng)建子進(jìn)程,創(chuàng)建的子進(jìn)程將會(huì)得到父進(jìn)程的數(shù)據(jù)空間、堆、棧......副本(采用寫時(shí)復(fù)制機(jī)制),子進(jìn)程將會(huì)繼承父進(jìn)程的信號(hào)掩碼、信號(hào)處理方式、當(dāng)前工作目錄、會(huì)話id、組id......。當(dāng)子進(jìn)程退出時(shí)父進(jìn)程應(yīng)當(dāng)及時(shí)獲取子進(jìn)程退出狀態(tài),否則,如果父進(jìn)程是一直在運(yùn)行,那么子進(jìn)程的退出狀態(tài)將一直保存在內(nèi)存中,直到父進(jìn)程退出才釋放。
我們可以使用如下幾種方法避免僵尸進(jìn)程的產(chǎn)生:
1.在fork后調(diào)用wait/waitpid函數(shù)取得子進(jìn)程退出狀態(tài)。
2.調(diào)用fork兩次(第一次調(diào)用產(chǎn)生一個(gè)子進(jìn)程,第二次調(diào)用fork是在第一個(gè)子進(jìn)程中調(diào)用,同時(shí)將父進(jìn)程退出(第一個(gè)子進(jìn)程退出),此時(shí)的第二個(gè)子進(jìn)程的父進(jìn)程id為init進(jìn)程id(注意:新版本Ubuntu并不是init的進(jìn)程id))。
3.在程序中顯示忽略SIGCHLD信號(hào)(子進(jìn)程退出時(shí)會(huì)產(chǎn)生一個(gè)SIGCHLD信號(hào),我們顯示忽略此信號(hào)即可)。
4.捕獲SIGCHLD信號(hào)并在捕獲程序中調(diào)用wait/waitpid函數(shù)。
方法一:
#include "../common/common.h" int main(void) { pid_t pid; if ((pid = fork()) < 0) { perror("fork error"); return EXIT_FAILURE; } else if (0 == pid) { printf("[%ld] child process is running...\n", (long)getpid()); _exit(0); } //sleep(15); if (waitpid(pid, NULL, 0) < 0) { perror("waitpid error"); return EXIT_FAILURE; } for (; ;) { pause(); } return EXIT_SUCCESS; }
方法二:
#include <sys/wait.h> #include "../common/common.h" int main(void) { pid_t pid; if ((pid = fork()) < 0) { perror("fork error"); return EXIT_FAILURE; } else if (0 == pid) { printf("first child is running..\n"); /**在第一個(gè)子進(jìn)程中再次fork***/ if ((pid = fork()) < 0) { perror("fork error"); return EXIT_FAILURE; } else if (pid > 0) {/**父進(jìn)程退出**/ printf("[%ld] first child is exit...\n", (long)getpid()); _exit(0); } sleep(2);/**確保父進(jìn)程先運(yùn)行**/ printf("second process pid: %ld, second process's parent pid: %ld\n", (long)getpid(), (long)getppid()); //sleep(15); printf("[%ld] is exit..\n", (long)getpid()); _exit(0); } /***獲得第一個(gè)子進(jìn)程的退出狀態(tài)***/ if (waitpid(pid, NULL, 0) < 0) { perror("waitpid error"); return EXIT_FAILURE; } for(;;) pause(); return EXIT_SUCCESS; }
方法三:
#include <signal.h> #include "../common/common.h" int main(void) { /***顯示忽略SIGCHLD信號(hào)****/ if (signal(SIGCHLD, SIG_IGN) == SIG_ERR) { perror("signal error"); return EXIT_SUCCESS; } pid_t pid; int i; /**產(chǎn)生10個(gè)子進(jìn)程***/ for (i=0; i<10; ++i) { if ((pid = fork()) < 0) { perror("fork error"); return EXIT_FAILURE; } else if (0 == pid) { _exit(0); } sleep(2); continue; } for (; ;) pause(); return EXIT_SUCCESS; }
方法四:
#include <signal.h> #include <sys/wait.h> #include "../common/common.h" void sig_chld(int signo); int main(void) { /**捕獲此信號(hào), 此刻系統(tǒng)會(huì)立刻檢測(cè)是否有次信號(hào)產(chǎn)生**/ if (signal(SIGCHLD, sig_chld) == SIG_ERR) { handler_err("signal error to SIGCHLD"); } pid_t pid; int i; for (i=0; i<10; i++) { if ((pid = fork()) < 0) { handler_err("fork error"); } else if (0 == pid) { printf("child pid: %d\n", getpid()); _exit(0); } sleep(1); continue; } for (; ;) { pause(); } return EXIT_SUCCESS; } /**捕獲到信號(hào)后會(huì)立刻執(zhí)行此段代碼***/ void sig_chld(int signo) { printf("receive child signal\n"); if (waitpid(-1, NULL, 0) < 0) { perror("waitpid error"); } if (signal(SIGCHLD, sig_chld) == SIG_ERR) { perror("signal error to SIGCHLD"); } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Git 刪除遠(yuǎn)程服務(wù)器文件同時(shí)保留本地文件實(shí)例詳解
這篇文章主要介紹了Git 刪除遠(yuǎn)程服務(wù)器文件同時(shí)保留本地文件實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2017-05-05linux環(huán)境搭建圖數(shù)據(jù)庫(kù)neo4j的講解
今天小編就為大家分享一篇關(guān)于linux環(huán)境搭建圖數(shù)據(jù)庫(kù)neo4j的講解,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-04-04Windows10安裝linux子系統(tǒng)的兩種方式(圖文詳解)
這篇文章主要介紹了Windows10安裝linux子系統(tǒng)的兩種方式,文中通過(guò)圖文介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06詳解Linux LVM邏輯卷配置過(guò)程(創(chuàng)建,增加,減少,刪除,卸載)
這篇文章主要介紹了詳解Linux LVM邏輯卷配置過(guò)程(創(chuàng)建,增加,減少,刪除,卸載),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08用DNSPod和Squid打造自己的CDN (六) 編譯并安裝Squid
這篇文章主要介紹centos下編譯并安裝Squid的方法,需要的朋友可以參考下2013-04-04