利用SIGCHLD信号回收子进程

子进程执行结束之后,父进程如果不对其进行回收,子进程就会变为僵尸进程。
父进程可以通过调用wait()函数和waitpid()函数去回收子进程.
由于子进程结束时会发送SIGCHLD信号给父进程,不过此信号的默认动作为忽略,我们可以通过系统函数sigaction()设置信号捕捉,在信号捕捉函数中去回收子进程。
信号捕捉函数中回收子进程的方法:
waitpid函数

下面展示一些 代码片

#include <stdio.h>
#include <unistd.h>
#include <signal.h>
#include <sys/wait.h>
#include <stdlib.h>void cat_sigchild(int sig)
{int status;pid_t pid;while((pid =  waitpid(0,&status,WNOHANG)) > 0){if(WIFEXITED(status)){printf("child exit with %d\n",WEXITSTATUS(status));}else if(WIFSIGNALED(status)){printf("child kill by sig %d\n",WTERMSIG(status));}}
}int main(int argc,char *argv[])
{pid_t pid;int i;for(i=0;i<10;i++){pid = fork();if(pid == 0){break;}else  if(pid <0){perror("fork:");exit(1);}}if(pid == 0)
{//if(i== 5)//sleep(i+1);printf("i am %d child pid is %d\n",i+1,getpid());
return i+1;
}
else if(pid >0)
{int n = 10000000;
//  sleep(4);//阻塞SIGCHLD信号sigset_t newset,oldset;sigemptyset(&newset);sigaddset(&newset,SIGCHLD);sigprocmask(SIG_BLOCK,&newset,&oldset); //如果在信号捕捉函数设置之前,子进程先结束,那么如果提前不阻塞SIGCHLD信号的话,就会导致先结束的子进程无法回收,所以在设置信号捕捉函数之前应该先阻塞SIGCHLD信号。while(n --)  //模拟程序执行,让子进程先结束。{}struct sigaction act,oldact;act.sa_handler = cat_sigchild;act.sa_flags = 0;sigemptyset(&act.sa_mask);sigaction(SIGCHLD,&act,NULL);//解除阻塞sigprocmask(SIG_SETMASK,&oldset,NULL);  while(1){printf("parent id %d\n",getpid());sleep(1);}
}
return 0;
}

注意点:
这里需要注意的是:我们在设置SIGCHLD信号的信号捕捉函数之前为了程序的严谨性,要先使用系统函数sigprocmask()去阻塞SIGCHLD信号,在设置完SIGCHLD信号的信号捕捉函数之后再解除阻塞。
原因:
如果我们的子进程先于父进程执行,假如在父进程设置完SIGCHLD的信号捕捉函数之前所有子进程都执行结束了,那么父进程就不会再收到子进程发送的SIGCHLD信号,信号捕捉函数就不会执行,进而回收子进程的系统函数waitpid()就不会被调用,那么就会造成所有的子进程变为僵尸进程。

解决办法:

1:设置信号阻塞,在父进程设置完SIGCHLD信号的信号捕捉函数之前阻塞SIGCHLD信号。这样即使所有的子进程都结束了,由于信号处于阻塞状态,当我们设置完SIGCHLD信号的捕捉函数之后,解除阻塞,就可以正常捕获到子进程发送的SIGCHLD信号,就可以进入捕捉函数调用waitpid去回收 死去的子进程 的资源。
注意
这里想要回收所有子进程调用waitpid必须使用while循环结构,不能使用if结构。因为在执行SIGCHLD信号捕捉函数期间,如果两个或多个子进程同时结束,那么SIGCHLD信号只记录一次,此时如果使用if结构就会导致同时结束的子进程只回收一个。

2:我们可以人为的使用kill -17 (父进程号) 命令给父进程发送SIGCHLD信号,就会触发信号捕捉函数,在捕捉函数内部利用
while((pid = waitpid(0,&status,WNOHANG)) > 0)结构去回收所有死掉的子进程。
注意:在父进程之前死掉的子进程发送的SIGCHLD信号默认动作为忽略,所以可以将之前先于父进程死掉的子进程回收靠的并不是之前子进程的SIGCHLD信号,而是靠的后来收到的SIGCHLD信号,和waitpid函数.
在这种人为发送信号回收的情况下:
SIGCHLD信号捕捉函数只调用执行一次,waitpid函数执行多次,才可以将之前死掉的所有子进程回收。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/384726.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【汇编语言】8086汇编语言的debug中,t和p命令的区别

单步跟踪命令&#xff08;单步进入(step in)&#xff09;&#xff1a;t命令是单步执行&#xff0c;是真的单步执行&#xff0c;碰到任何地方&#xff0c;都会一步步执行&#xff0c;遇到子程序&#xff0c;也会进入里面一步步执行再返回 单步执行命令&#xff08;单步通过(ste…

哲学家就餐问题(如何避免死锁)(多线程版)

哲学家就餐问题 多线程编程中&#xff0c;常常会遇到线程间访问共享资源的问题&#xff0c;如果处理不当则会发生死锁&#xff0c;某一个线程可能永远访问不到共享资源。 为了避免死锁的发生&#xff0c;提出哲学家就餐问题。 下面展示一些代码片段 #include <stdio.h>…

【汇编语言】汇编语言如何编程,如何避免出错

0 前言 这里以8086汇编语言举例。 1 重要方法&#xff1a;画流程图 这一点不再多说&#xff01;你懂的&#xff01;只要真的画流程图&#xff0c;并且按此翻译为汇编代码&#xff0c;程序出错几率就会减少很多。 2 框架思维&#xff1a;构建标准编程框架 这里只举例说明一…

【汇编语言】8086汇编字符串定义为何使用DB?其他数据类型不可以吗?(20200515复盘)

目录0 前言0.1 先告诉你结论1 8086汇编语言中的字符串1.1 字符串的定义与使用1.2 直接定义的细节1.2.1 使用DB数据类型1.2.2 使用其他数据类型1.3 直接使用的细节1.3.1 赋值给寄存器1.3.2 赋值给内存单元1.3.3 字符串直接使用的本质1.4 应用2 扩展&#xff1a;数据定义的方式3 …

哲学家就餐(避免死锁)(多进程版)

哲学家就餐&#xff08;避免死锁&#xff09;&#xff08;多进程版&#xff09; 哲学家就餐利用信号量在多进程之间实现 下面展示一些代码片段 #include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <sys/wait.h> #include <sys…

【汇编语言】数据类型的匹配问题:自动匹配与手动匹配

0 前言 本文基于8086汇编语言&#xff0c;不过x86系列语言具备向下兼容特点&#xff0c;大多数情况都可以用。 与高级语言数据类型的自动转换和强制转换一样&#xff0c;汇编语言的数据类型也有 自动匹配和手动匹配。 下面&#xff0c;我来介绍一下汇编语言的相关原则。 1 …

生产者消费者模型(条件变量+互斥锁)

生产者消费者模型 线程同步典型的案例即为生产者消费者模型&#xff0c;而借助条件变量来实现这一模型&#xff0c;是比较常见的一种方法。假定有两个线程&#xff0c;一个模拟生产者行为&#xff0c;一个模拟消费者行为。两个线程同时操作一个共享资源&#xff0c;生产者向其…

【汇编语言】王爽实验8,分析一个奇怪的程序,学习笔记(20200517)

0 前言 先给出源程序 assume cs:code code segmentmov ax,4c00hint 21hstart:mov ax,0s:nopnopmov di,offset smov si,offset s2mov ax,cs:[si]mov cs:[di],axs0:jmp short ss1:mov ax,0int 21hmov ax,0s2:jmp short s1nopcode ends end start运行结果&#xff1a;本程序能够…

Qt制作定时关机小程序

文章目录 完成效果图ui界面ui样图 main函数窗口文件头文件cpp文件 引言 一般定时关机采用命令行模式&#xff0c;还需要我们计算在多久后关机&#xff0c;我们可以做一个小程序来定时关机 完成效果图 ui界面 <?xml version"1.0" encoding"UTF-8"?>…

MIPS快速入门(原文+翻译):MIPS Architecture and Assembly Language Overview(持续更新中)

前言 发布该文章的网站已经无法访问&#xff0c;无法获得相关翻译授权&#xff0c;本人的翻译仅供大家参考学习&#xff0c;尽可能使用直译&#xff0c;并加上一些译者注&#xff08;使用“ [1] ”的形式&#xff09;&#xff0c;以减少信息损失&#xff0c;水平有限&#xff…

Visual Studio 编译优化选项:Debug与Release、禁止优化与O1、O2、Ox优化

Debug与禁止优化 Debug模式是调试模式&#xff0c;会有很多冗余的调试代码&#xff0c;供开发者调试程序使用。 VS是默认使用Debug模式的&#xff0c;我使用的是VS 2017。 在Debug模式下&#xff0c;是默认开启禁止优化的&#xff0c;我们来查看一下 在左侧源文件的main.c处…

【汇编语言】记录一组数中负数的个数,8086与MIPS汇编程序

题目及解答 统计由DATA开始的字节数据串中负元素的个数&#xff0c;数据个数在COUNT单元&#xff0c;统计结果存入RLT单元。 8086汇编&#xff1a; ; 统计数字中负数的个数【循环中加了个if else】 assume ds:datasg datasg segmentdata db 1,-2,-3,-1,-4,0,-2 count dw 7 ; 数…

【数字逻辑入门】计算机如何存储1位二进制数

0 前言 本文将会以R-S锁存器为例&#xff0c;引出锁存器的核心和本质&#xff0c;之后再带你构建更多类型的锁存器&#xff0c;你能够&#xff1a; 感受到由浅入深的学习方式体会到掌握核心本质的快感深刻理解核心套外壳的设计理念&#xff08;产品迭代1.0–>2.0–>3.0…

【算法训练】DAY1:整数反转

1 前言 题目来源于Leetcode。 重点&#xff1a;理清逻辑&#xff0c;忽略细节&#xff0c;模仿高手&#xff0c;五毒神掌 2 题目分析 题目很容易理解&#xff0c;先分成两个部分 正数负数 先解决正数 最开始想到的是 intchar数组long唯一增加的就是&#xff0c;先判断整…

【汇编语言】(x86)test与跳转指令(je jle jge jg jl……)组合的含义

在x86指令集中&#xff0c;经常遇到test指令与条件跳转指令组合&#xff0c;这是什么含义呢&#xff1f; 博主表示&#xff0c;查了很多资料也没人完全说清楚…… 这里只用最简单的&#xff0c;抽象层次进行说明&#xff0c;不讲原理。 举例 test edx,edx jle 某地址含义是&…

【蓝桥杯】BASIC-8 回文数(2020-06-08)

题目 试题 基础练习 回文数 资源限制 时间限制&#xff1a;1.0s 内存限制&#xff1a;512.0MB 问题描述   1221是一个非常特殊的数&#xff0c;它从左边读和从右边读是一样的&#xff0c;编程求所有这样的四位十进制数。    输出格式   按从小到大的顺序输出满足条件的…

【算法训练】Leetcode 1295. 统计位数为偶数的数字(2020.06.09 )

1 题目 1295. 统计位数为偶数的数字 给你一个整数数组 nums&#xff0c;请你返回其中位数为 偶数 的数字的个数。 示例 1&#xff1a; 输入&#xff1a;nums [12,345,2,6,7896] 输出&#xff1a;2 解释&#xff1a; 12 是 2 位数字&#xff08;位数为偶数&#xff09; 345 …

Vivado设置指定源文件进行RTL优化

像VS编译器设置启动项一样&#xff0c;Vivado中&#xff0c;也有类似设计&#xff0c;可以看到&#xff0c;当前选中的是ALU&#xff0c;那么进行RTL优化的时候&#xff0c;会优化RTL的结果&#xff0c;而不是别的&#xff0c;如何改成别的&#xff1f; 在某文件上右键单击选择…

【完整流程】用VSCode替换Vivado默认编辑器

本文楼主找了很多资料&#xff0c;选出了最有用的资料&#xff0c;按照教程走&#xff0c;就可以顺利搞定&#xff0c;先给出画面 很酷很方便&#xff0c;同时还有 自动补全检测错误列选自动生成仿真测试文件 等重要功能 Vivado原来的编辑器是这样的…… 关键是&#xff0c…

IEDA中JavaDoc的自动生成、手动生成,以及生成html文档

1 自动生成类的注释 JavaDoc就是java特有的一种注释。 1.1 配置 首先&#xff0c;IDEA点击File-->Settings 然后Editor-->File and Code Templates-->Class 之后在这地方&#xff0c;添加一些代码 /** * ${description} * * <p> * 创建日期&#xff1a;$…