SIGCHLD信号

在进程的学习中,我们用wait和waitpid函数清理僵尸进程,父进程可以阻塞式等待子进程结束,也可以非阻塞地查询是否有子进程结束等待清理(即轮询方式)。采用第一种方式,父进程阻塞式等待,就不能处理自己的工作了;采用第二种方式,父进程在处理自己工作的同时还要记得时不时地轮询一下,程序实现复杂。

其实,子进程在终止时会给父进程SIGCHLD信号,该信号的默认处理动作是忽略,父进程可以自定义SIGCHLD信号的处理函数,这样父进程只需专心处理自己的工作,不不关心子进程了,子进程终止时会通知父进程,父进程在信号处理函数中调用wait清理子进程即可。

事实上,由于Unix的历史原因,要想不产生僵尸进程还有另外一种方法:父进程调用sigaction将SIGCHLD的处理动作置为SIG_IGN,这样fork出来的子进程在终止时会自动清理掉,不会产生僵尸进程,也不会通知父进程。系统默认的忽略函数和用户用sigaction函数自定义的忽略,通常是没有区别的,但这是一个特例。此方法对于Linux可用,但不保证在其他Unix系统上都可用。

  • 代码实例:

要求代码实现以下功能:父进程fork出子进程,子进程调用exit(2)终止,父进程自定义SIGCHLD信号的处理函数,在其中调用wait获得子进程的退出状态并打印。

#include <stdio.h>
#include<stdlib.h>
#include<signal.h>

void handler(int sig){
    pid_t id;
    while((id = waitpid(-1,NULL,WNOHANG)) > 0){
            printf("wait child success:%d\n",id);
    }
    printf("child is quit!%d\n",getpid());
}

int main(){
    signal(SIGCHLD,handler);
    pid_t cid;
    if((cid = fork()) == 0){//child
        printf("child:%d\n",getpid());
        sleep(3);
        exit(1);
    }
    while(1){
        printf("father proc is doing something!\n");
        sleep(1);
    }
    return 0;
}

运行结果:

     


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值