在UNIX 系统中,一个进程结束了,但是他的父进程没有等待(调用wait / waitpid)他, 那么他将变成一个僵尸进程. 在fork()/execve()过程中,假设子进程结束时父进程仍存在,而父进程fork()之前既没安装SIGCHLD信号处理函数调用 waitpid()等待子进程结束,又没有显式忽略该信号,则子进程成为僵尸进程。
两种方法杀死僵尸进程,一般僵尸进程很难直接kill掉,不过您可以kill僵尸父进程。父进程死后,僵尸进程成为”孤儿进程”,过继给1号进程init,init始终会负责清理僵尸进程.它产生的所有僵尸进程也跟着消失。
(1)利用wait,waitpid函数来主动请求子进程的函数
* wait,waitpid函数
pid_t wait(int *status);
pid_t waitpid(pid_t pid, int *status, int options);
(2)利用sigaction函数进行信号处理,通过注册子进程触发的SIGCHLD信号来
编写相应的处理函数。
* 信号sigaction函数和sigaction结构体结构
int sigaction(int sig, const struct sigaction *restrict act, struct sigaction *restrict oact);
struct sigaction
{
void (*sa_handler)(int);//传统信号处理函数
void (*sa_sigaction)(int, siginfo_t *, void *);//新型信号函数处理指针
sigset_t sa_mask;//信号屏蔽集
int sa_flags;//一般设置为0
void (*sa_restorer)(void);//保留不要使用
};
act是新的信号处理函数,oact是旧的处理函数一般为NULL
* 代码案例如下:
#include<iostream>
#include<sys/wait.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<cstdio>
#include<cstdlib>
#include<cstring>
void handle(int signum)
{
sleep(100);
int status;
pid_t pid=waitpid(-1,&status,WNOHANG);/*status是子进程返回的值*/
//pid_t pid=waitpid(-1,NULL,WNOHANG);就可以回收全部僵尸进程
if(WIFEXITED(status))
{
std::cout<<"recover child id "<<pid<<std::endl;/*回收的僵尸进程ID*/
std::cout<<"child send:"<<WEXITSTATUS(status)<<std::endl;/*僵尸进程退出的值*/
}
}
int main(int argc,char** argv)
{
pid_t ch_id;
struct sigaction act;
bzero(&act,sizeof(act));
act.sa_handler=handle;
sigemptyset(&act.sa_mask);
act.sa_flags=0;
sigaction(SIGCHLD,&act,NULL);
int num=10;
while(num)
{
if(fork()==0)
{
std::cout<<"I'm zombie child process"<<std::endl;
_exit(num);
}
std::cout<<"wait!"<<std::endl;
sleep(2);
num--;
}
}
其实最直接的方法下面代码就可以做到回收
void handle_child()
{
struct sigaction act;
bzero(&act,sizeof(act));
act.sa_handler=SIG_IGN;
sigemptyset(&act.sa_mask);
act.sa_flags=0;
sigaction(SIGCHLD,&act,NULL);
}
子进程终止的时候产生SIGCHLD信号,系统会用SIG_IGN函数来进行默认处理,
所以正常情况下只需要设置上面的代码就可以了。