SIGCHLD的产生条件

子进程终止时

子进程接收到SIGSTOP信号停止时

子进程处在停止态,接受到SIGCONT后唤醒时

也就是说:子进程的运行状态发生变化就会发送SIGCHILD信号;这里的意思时,子进程比较依恋父母,自己发生变化就要给父母说一下。

借助SIGCHLD信号回收子进程

子进程结束运行,其父进程会收到SIGCHLD信号。该信号的默认处理动作是忽略。可以捕捉该信号,在捕捉函数中完成子进程状态的回收。不多说,上代码:

#include
<stdlib.h>

#include
<stdio.h>

#include
<sys/types.h>

#include
<sys/wait.h>

#include
<unistd.h>

#include
<signal.h>

void sys_err(char *p)

{

    perror(p);

    exit(1);

}

void do_sig_child(int
signal)

{

    printf("++++++++++++++++我是子进程,我的ID = %d\n", getpid());

}

void do_sig_parent(int
signal)

{

    int status;

    while ((waitpid(0, &status, WNOHANG)>0))//如果清理成功

    {

        if (WIFEXITED(status))

        {

            printf("子进程正常退出,状态:%d\n", WEXITSTATUS(status));

        }

        else
if (WIFSIGNALED(status))

        {

            printf("———子进程异常退出,状态:%d\n", WTERMSIG(status));

        }

    }

}

int main(void)

{

    int i = 0;

    pid_t pid;

    //屏蔽信号SIGCHLD

    for (; i != 10; i++)

    {

        if (!(pid = fork()))

        {

            break;

        }

        else
if (pid<0)

        {

            sys_err("fork error ");

        }

    }

    if (i<10)

    {

        do_sig_child(0);

        sleep(1);//防止在父进程还未注册信号捕捉函数的时候子进程死亡。这样父进程就不能回收已死亡的子进程。

    }

    else

    {

        struct
sigaction act, oldact;

        //修改信号处理动作

        act.sa_handler = do_sig_parent;

        act.sa_flags = 0;

        sigemptyset(&act.sa_mask);

        sigaddset(&act.sa_mask, SIGCHLD);

        if ((sigaction(SIGCHLD, &act, &oldact))<0)

        {

            sys_err("sigaction error");

        }

        while (1)

        {

            printf("父进程的ID = %d\n", getpid());

            sleep(1);

        }

        //还原信号处理动作

        if ((sigaction(SIGCHLD, &oldact, &act))<0)

        {

            sys_err("sigaction error");

        }

    }

    return 0;

}

现在,一切正常,运行结果:

现在我们来提问:如果每创建一个子进程后不使用sleep可以吗?可不可以将程序中,捕捉函数内部的while替换为if?为什么?

第一个问题我在代码中的注释已经写的很明白了,这里就不再说了。我们来说说另一个问题:捕捉函数do_sig_parent内部的while替换为if?为什么?即将代码 while((waitpid(0,&status,WNOHANG)>0))换为if((waitpid(0,&status,WNOHANG)>0))可以嘛?子进程一死亡就发送SIGCHLD信号给父进程,父进程收到后就执行捕捉函数。怎么不行呢?不管他行不行,我们试试就知道,实践出真知;结果如下:这次还有一个子进程未回收,不行了,大野的,这个win10子系统太牛,同样的代码在别人的Linux系统上错误能够完美重现,在这里能够完美避开。我试了好几十次才找到这一个错误。但是没事,反正有了。

来分析一下为啥if就不能完美的回收所有子进程:if只执行一次回收(因为到了那儿说明父进程收到信号了,也就是说一定有子进程死掉了。)但是若是某一个时刻有多个子进程同时死了,那么依据信号不进行排队的原理,就只有其中一个信号被处理。其他的被忽略。但是while就不一样了,就算同时来多个信号,但是while时循环执行,虽然信号处理先后顺序不一样,但是每个信号都会被处理啊(非阻塞的waitpid函数)。所以啊,当有多个子进程时,回收的时候用循环,不要用if。

SIGCHLD信号的更多相关文章

  1. unp TCP 客户端服务器回射程序中对SIGCHLD信号的处理

    第五章中,有一个例子模拟客户端并发的终止TCP连接,服务器捕捉并处理SIGCHLD信号并调用waitpid函数防止僵死进程的出现.信号处理函数中核心的一句是: , &statloc, WNOH ...

  2. linux下的僵尸进程处理SIGCHLD信号

    什么是僵尸进程? 首先内核会释放终止进程(调用了exit系统调用)所使用的所有存储区,关闭所有打开的文件等,但内核为每一个终止子进程保存了一定量的信息.这些信息至少包括进程ID,进程的终止状态,以及该 ...

  3. [转] linux下的僵尸进程处理SIGCHLD信号

    什么是僵尸进程? 首先内核会释放终止进程(调用了exit系统调用)所使用的所有存储区,关闭所有打开的文件等,但内核为每一个终止子进程保存了一定量的信息.这些 信息至少包括进程ID,进程的终止状态,以及 ...

  4. 三十一、Linux 进程与信号——SIGCHLD 信号、kill和raise函数以及alarm函数

    31.1 SIGCHLD 信号 子进程状态发生变化(子进程结束)产生该信号,父进程需要使用 wait 调用来等待子进程结束并回收它. 避免僵尸进程 #include <stdio.h> # ...

  5. linux下的僵尸进程处理SIGCHLD信号【转】

    转自:http://www.cnblogs.com/wuchanming/p/4020463.html 什么是僵尸进程? 首先内核会释放终止进程(调用了exit系统调用)所使用的所有存储区,关闭所有打 ...

  6. 僵尸进程与SIGCHLD信号

    什么是僵尸进程? 首先内核会释放终止进程(调用了exit系统调用)所使用的所有存储区,关闭所有打开的文件等,但内核为每一个终止子进程保存了一定量的信息.这些信息至少包括进程ID,进程的终止状态,以及该 ...

  7. 2次使用fork避免产生僵尸进程和不去处理SIGCHLD信号

    1.如下代码所示 #include <unistd.h> #include <sys/types.h> #include <unistd.h> int main(i ...

  8. 僵进程与SIGCHLD信号

    参考: https://www.cnblogs.com/webor2006/p/4014586.html wait()和waitpid()的参数解析:https://blog.csdn.net/csd ...

  9. linux有关信号的FAQ

    1.为什么会出现系统调用被中断的情况? 进程在执行一个低速系统调用而阻塞期间捕捉到一个信号时,该系统调用就被中断不再继续执行.该系统调用返回出错,其errno被设置为EINTR.这样处理的理由是:因为 ...

随机推荐

  1. jquery.autocomplete 搜索文字提示

    function GetJobTitle(obj) { $(obj).autocomplete("GetJobTitle.ashx", { max: 12, //列表里的条目数 m ...

  2. [UE4]UMG和关卡坐标变换、旋转小地图

    一.优化上一节的蓝图,新建一个函数addFlagToCanvas(动态添加图标到Canvas) 二. 分析地图坐标系和UMG坐标系 要根据实际情况分析关卡坐标系. UserWidget中的坐标系 三. ...

  3. Nginx 分布式session共享问题

    在集群的时候每次访问,都会被代理转到不同的服务器,那么在这些服务器之间如何共享session? 解决方式1:session复制 只能在window下好使,web服务器解决(广播机制,将一台机器上的se ...

  4. C语言:传值,传地址

    形参:形式参数实参:实际参数 传值: 把实参的值复制给形参, 修改函数内的形参,不会影响实参. 传地址: 指针传值,形参为指向实参地址的指针 当对形参的指向操作时,相当于对实参本身进行的操作 #inc ...

  5. 基于Linux命令行KVM虚拟机的安装配置与基本使用

    背景 由于生产环境的服务器并不会安装桌面环境,简单操作的图形化安装也不适合批量部署安装.因此,我还是更倾向于在命令下安装配置KVM虚拟机.结合了一些资料和个人使用的状况,我大致列出了一些基本和常用的使 ...

  6. Nginx 之防盗链配置

    首先,我们需要知道通过什么来实现防盗的! http referer 是header的一部分,当浏览器向web服务器发送请求的时候,一般会带上referer,这是在告诉服务器是从哪个页面链接过来的,服务 ...

  7. 初级安全入门——XSS注入的原理与利用

    XSS的简单介绍 跨站脚本攻击(Cross Site Scripting),为不和层叠样式表(Cascading Style Sheets,CSS)的缩写混淆,故将跨站脚本攻击缩写为XSS.恶意攻击者 ...

  8. sqlserver表数据的修改

    清除表数据  truncate table  [表名称] 将表b中的一列数据,更新到表tableA  如: tableA .key tableA .value 123   124   tableB.k ...

  9. 零基础学习python_类和对象(36-40课)

    今天我们开始学习面向对象的知识咯,之前我对面向对象也学的懵懵的,因为感觉知道好像又不是特别清楚,接下来我们一起来学习类和对象吧.零基础的课程我都是看小甲鱼的视频学的,没基础的可以去这个网址下载视频学习 ...

  10. 认识下java注解的实现原理

    1,什么是注解 注解也叫元数据,例如常见的@Override和@Deprecated,注解是JDK1.5版本开始引入的一个特性,用于对代码进行说明,可以对包.类.接口.字段.方法参数.局部变量等进行注 ...