详解linux下避免僵尸进程的几种方法
linux下我们可以调用fork函数创建子进程,创建的子进程将会得到父进程的数据空间、堆、栈......副本(采用写时复制机制),子进程将会继承父进程的信号掩码、信号处理方式、当前工作目录、会话id、组id......。当子进程退出时父进程应当及时获取子进程退出状态,否则,如果父进程是一直在运行,那么子进程的退出状态将一直保存在内存中,直到父进程退出才释放。
我们可以使用如下几种方法避免僵尸进程的产生:
1.在fork后调用wait/waitpid函数取得子进程退出状态。
2.调用fork两次(第一次调用产生一个子进程,第二次调用fork是在第一个子进程中调用,同时将父进程退出(第一个子进程退出),此时的第二个子进程的父进程id为init进程id(注意:新版本Ubuntu并不是init的进程id))。
3.在程序中显示忽略SIGCHLD信号(子进程退出时会产生一个SIGCHLD信号,我们显示忽略此信号即可)。
4.捕获SIGCHLD信号并在捕获程序中调用wait/waitpid函数。
方法一:
#include"../common/common.h" intmain(void) { pid_tpid; if((pid=fork())<0){ perror("forkerror"); returnEXIT_FAILURE; }elseif(0==pid){ printf("[%ld]childprocessisrunning...\n",(long)getpid()); _exit(0); } //sleep(15); if(waitpid(pid,NULL,0)<0){ perror("waitpiderror"); returnEXIT_FAILURE; } for(;;){ pause(); } returnEXIT_SUCCESS; }
方法二:
#include#include"../common/common.h" intmain(void) { pid_tpid; if((pid=fork())<0){ perror("forkerror"); returnEXIT_FAILURE; }elseif(0==pid){ printf("firstchildisrunning..\n"); /**在第一个子进程中再次fork***/ if((pid=fork())<0){ perror("forkerror"); returnEXIT_FAILURE; }elseif(pid>0){/**父进程退出**/ printf("[%ld]firstchildisexit...\n",(long)getpid()); _exit(0); } sleep(2);/**确保父进程先运行**/ printf("secondprocesspid:%ld,secondprocess'sparentpid:%ld\n",(long)getpid(),(long)getppid()); //sleep(15); printf("[%ld]isexit..\n",(long)getpid()); _exit(0); } /***获得第一个子进程的退出状态***/ if(waitpid(pid,NULL,0)<0){ perror("waitpiderror"); returnEXIT_FAILURE; } for(;;) pause(); returnEXIT_SUCCESS; }
方法三:
#include#include"../common/common.h" intmain(void) { /***显示忽略SIGCHLD信号****/ if(signal(SIGCHLD,SIG_IGN)==SIG_ERR){ perror("signalerror"); returnEXIT_SUCCESS; } pid_tpid; inti; /**产生10个子进程***/ for(i=0;i<10;++i){ if((pid=fork())<0){ perror("forkerror"); returnEXIT_FAILURE; }elseif(0==pid){ _exit(0); } sleep(2); continue; } for(;;) pause(); returnEXIT_SUCCESS; }
方法四:
#include#include #include"../common/common.h" voidsig_chld(intsigno); intmain(void) { /**捕获此信号,此刻系统会立刻检测是否有次信号产生**/ if(signal(SIGCHLD,sig_chld)==SIG_ERR){ handler_err("signalerrortoSIGCHLD"); } pid_tpid; inti; for(i=0;i<10;i++){ if((pid=fork())<0){ handler_err("forkerror"); }elseif(0==pid){ printf("childpid:%d\n",getpid()); _exit(0); } sleep(1); continue; } for(;;){ pause(); } returnEXIT_SUCCESS; } /**捕获到信号后会立刻执行此段代码***/ voidsig_chld(intsigno) { printf("receivechildsignal\n"); if(waitpid(-1,NULL,0)<0){ perror("waitpiderror"); } if(signal(SIGCHLD,sig_chld)==SIG_ERR){ perror("signalerrortoSIGCHLD"); } }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。