僵尸进程的处理

在UNIX系统中,未被父进程等待的已结束进程会变成僵尸进程。避免僵尸进程的方法包括使用wait、waitpid函数请求子进程状态,或者通过sigaction函数处理SIGCHLD信号。当父进程被kill后,子进程会由init接管并清理。示例代码展示了如何通过信号处理函数回收僵尸进程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在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函数来进行默认处理,
所以正常情况下只需要设置上面的代码就可以了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值