【进程信号拓展】SIG_CHLD 信号处理

SIGCHLD信号

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

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

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
void handler(int signo)
{
    pid_t id = waitpid(-1, NULL, WNOHANG); // 使用WNOHANG才能保证非阻塞查询
    while(id > 0) 
    {
        printf("wait child success: %d\n", id);
    }
    printf("child is quit! %d\n", getpid());
}
int main()
{
    signal(SIGCHLD, handler); // 将SIGCHLD信号进行自定义捕捉
    pid_t cid = fork();
    if(cid == 0)
    {  
        // child
        printf("child : %d\n", getpid());
        sleep(3);
        exit(1);
    }

    // father
    while(1)
    {
        printf("father proc is doing some thing!\n");
        sleep(1);
    }
    return 0;
}

​ 事实上,由于 UNIX 的历史原因,要想不产生僵尸进程还有另外一种办法:父进程调用 sigactionSIGCHLD 的处理动作置为 SIG_IGN,这样 fork 出来的子进程在终止时会自动清理掉,不会产生僵尸进程,也不会通知父进程。

​ 💥💥💥 此方法对于 Linux 可用,但不保证在其它 UNIX 系统上都可用

​ 下面我们来编写程序验证这样做不会产生僵尸进程:

cpp 复制代码
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
int main()
{
    signal(SIGCHLD, SIG_IGN); // 直接通过SIG_IGN忽略,即可回收僵尸子进程
    pid_t cid;
    if((cid = fork()) == 0)
    {  
        // child
        printf("child : %d\n", getpid());
        sleep(3);
        exit(1);
    }

    // father
    while(1)
    {
        printf("father proc is doing some thing!\n");
        sleep(1);
    }
    return 0;
}

​ 除此之外,系统默认的忽略动作 Ign 和用户用 sigaction 函数自定义的忽略动作 SIG_IGN 通常是没有区别的,但这是一个特例 。也就是说,SIG_IGN 其实做了更多的一些小动作!

相关推荐
零号全栈寒江独钓1 小时前
基于c/c++实现linux/windows跨平台获取ntp网络时间戳
linux·c语言·c++·windows
左手厨刀右手茼蒿2 小时前
Linux 内核中的进程管理:从创建到终止
linux·嵌入式·系统内核
geinvse_seg2 小时前
中小团队如何低成本搭建项目管理系统?基于 Ubuntu 的 Dootask 私有化部署实战
linux·运维·ubuntu
CSCN新手听安2 小时前
【linux】高级IO,以ET模式运行的epoll版本的TCP服务器实现reactor反应堆
linux·运维·服务器·c++·高级io·epoll·reactor反应堆
丶伯爵式2 小时前
Ubuntu 24.04 更换国内软件源指南 | 2026年3月26日
linux·运维·ubuntu·国内源·升级
左手厨刀右手茼蒿2 小时前
Linux 内核中的 DMA 管理:从缓冲区到传输
linux·嵌入式·系统内核
Java后端的Ai之路2 小时前
Linux端口进程查找与终止教程
linux·运维·服务器
松☆3 小时前
C++ 算法竞赛题解:P13569 [CCPC 2024 重庆站] osu!mania —— 浮点数精度陷阱与 `eps` 的深度解析
开发语言·c++·算法
(Charon)3 小时前
【C++/Qt】C++/Qt 实现 TCP Server:支持启动监听、消息收发、日志保存
c++·qt·tcp/ip
北山有鸟4 小时前
【学习笔记】MIPI CSI-2 协议全解析:从底层封包到像素解析
linux·驱动开发·笔记·学习·相机