詳解linux下避免僵尸進(jìn)程的幾種方法
linux下我們可以調(diào)用fork函數(shù)創(chuàng)建子進(jìn)程,創(chuàng)建的子進(jìn)程將會(huì)得到父進(jìn)程的數(shù)據(jù)空間、堆、棧......副本(采用寫(xiě)時(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)文章
Ubuntu18.04安裝Pycharm教程的實(shí)現(xiàn)
這篇文章主要介紹了Ubuntu18.04安裝Pycharm教程的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09
CentOS7 systemd添加自定義系統(tǒng)服務(wù)的方法
這篇文章主要介紹了CentOS7 systemd添加自定義系統(tǒng)服務(wù),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-03-03

