SIGCHLD信号

1SIGCHLD信号产生的条件

1.子进程终止时会向父进程发送SIGCHLD信号,告知父进程回收自己,但该信号的默认处理动作为忽略,因此父进程仍然不会去回收子进程,需要捕捉处理实现子进程的回收;

2.子进程接收到SIGSTOP(19)信号停止时;

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

综上:子进程结束、接收到SIGSTOP停止(挂起)和接收到SIGCONT唤醒时都会向父进程发送SIGCHLD信号。父进程可以捕捉该信号,来实现对子进程的回收,或者了解子进程所处的状态。

2)借助SIGCHLD信号回收子进程

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

//分析例子:结合 17)SIGCHLD 信号默认动作,掌握父使用捕捉函数回收子进程的方法

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <signal.h>void sys_err(char *str)
{perror(str);exit(1);
}void do_sig_child(int signo)
{int status;pid_t pid;//    if ((pid = waitpid(0, &status, WNOHANG)) > 0) {while ((pid = waitpid(0, &status, WNOHANG)) > 0) {if (WIFEXITED(status))printf("------------child %d exit with %d\n", pid, WEXITSTATUS(status));else if (WIFSIGNALED(status))printf("child %d killed by the %dth signal\n", pid, WTERMSIG(status));}
}int main(void)
{pid_t pid;int i;//阻塞SIGCHLDfor (i = 0; i < 10; i++) {if ((pid = fork()) == 0)break;else if (pid < 0)sys_err("fork");}if (pid == 0) {     //10个子进程int n = 1;while (n--) {printf("child ID %d\n", getpid());sleep(1);}return i+1;      //子进程结束状态依次为1、2、••••••、10} else if (pid > 0) {   struct sigaction act;act.sa_handler = do_sig_child;sigemptyset(&act.sa_mask);act.sa_flags = 0;sigaction(SIGCHLD, &act, NULL);//解除对SIGCHLD的阻塞while (1) {printf("Parent ID %d\n", getpid());sleep(1);}}return 0;
}

[root@localhost 01_signal_test]# ./sigchild

child ID 11129

child ID 11130

child ID 11131

child ID 11132

child ID 11133

child ID 11134

child ID 11135

child ID 11136

child ID 11137

Parent ID 11128  //表明在此时父进程肯定完成了对信号的注册工作

child ID 11138

------------child 11129 exit with 1

------------child 11130 exit with 2

------------child 11132 exit with 4

------------child 11133 exit with 5

------------child 11131 exit with 3

Parent ID 11128

------------child 11134 exit with 6

------------child 11137 exit with 9

Parent ID 11128

------------child 11136 exit with 8

Parent ID 11128

------------child 11135 exit with 7

Parent ID 11128

------------child 11138 exit with 10 //可见父进程的确完成了对全部子进程的回收

Parent ID 11128

Parent ID 11128

分析如下:

1.在上述程序中,每个子进程在结束之前都会睡眠1s,这是为了留出足够的时间保证在所有子进程结束之前,父进程能够完成对SIGCHLD信号的注册。否则所有子进程都结束了,父进程还没有完成注册,则此时信号都被忽略,从而子进程不能被父进程回收。但是,只要有一个子进程在信号注册后结束,所有子进程都可以被回收,因为使用了while循环回收子进程。除了使用sleep函数来实现这一点外,其实更加高效而又精确的办法(其实在负载较大的 情况下,sleep函数也不能确保能够做到)就是在父进程注册函数之前就将SIGCHLD信号设置为阻塞(通过信号集操作函数加入到屏蔽字中),在注册完成时立即解除对该信号的阻塞即可。

2.不可以将捕捉函数内部的while替换为if。因为在执行捕捉函数期间,发送了多次SIGCHLD信号,未决信号集只是记录了一次,因此下一次再调用捕捉函数时,if只能完成对一个子进程的回收(即使有多个子进程都发了信号,但是只是调用一次捕捉函数)。而while循环则可以对所有结束了的子进程都完成回收。因此对于多个子进程的回收,最好采用循环的方式,不采用if。

3.之前在管道进程间通信,父子进程实现ls | wc -l时(父进程exec函数族执行ls,子进程exec执行wc -l),父进程无法完成对子进程的回收,因为父进程执行exec函数族就离开了,不能回来。但是现在可以利用信号捕捉实现,在父进程调用exec函数族之前就注册SIGCHLD的捕捉函数,但是需要注意以下几点:1.仍然要考虑对SIGCHLD信号的屏蔽和解除屏蔽操作;2.要考虑子进程向父进程发送信号时,父进程已经结束了,此时父进程仍然无法对子进程回收,可以让父进程睡眠等待,让其后结束(但这在实际中是没有意义的,父进程结束了,子进程变为孤儿进程,会被init进程回收,因此实际编程不需要考虑这一点);3.在ls | wc -l这个例子中,由于使用了标准输出的重定向(dup2),因此用户处理函数输出的回收状态信息不能显示在屏幕上,因此在用户处理函数中还需要再次进行重定向,恢复输出重定向为屏幕。

总结:

1.子进程继承了父进程的信号屏蔽字和信号处理动作,但子进程没有继承未决信号集spending。

2.注意注册信号捕捉函数的位置。

3.应该在fork之前,阻塞SIGCHLD信号。注册完捕捉函数后解除阻塞。

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

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

相关文章

信号传参

&#xff08;1&#xff09;发送信号传参 前面已经知道从一个进程向另一个进程发送信号可以使用kill函数&#xff0c;但是kill函数在向进程发送信号的时候不能携带除了信号以外的其他信息&#xff0c;这时可以使用与kill相对应的sigqueue函数&#xff0c;该函数也是向一个进程发…

【Leetcode | 52】257. 二叉树的所有路径

给定一个二叉树&#xff0c;返回所有从根节点到叶子节点的路径。 说明: 叶子节点是指没有子节点的节点。 示例: 输入: 1 / \ 2 3 \ 5 输出: ["1->2->5", "1->3"] 解释: 所有根节点到叶子节点的路径为: 1->2->5, 1->3 解法一&a…

623. 在二叉树中增加一行

给定一个二叉树&#xff0c;根节点为第1层&#xff0c;深度为 1。在其第 d 层追加一行值为 v 的节点。 添加规则&#xff1a;给定一个深度值 d &#xff08;正整数&#xff09;&#xff0c;针对深度为 d-1 层的每一非空节点 N&#xff0c;为 N 创建两个值为 v 的左子树和右子树…

终端的概念

操作系统接口&#xff1a;用户接口和程序接口。用户接口分为联机用户接口和脱机用户接口。脱机用户接口出现在早期的批处理系统中&#xff08;将作业提前交给操作系统&#xff0c;作业完成的过程中用户无法交互&#xff09;&#xff1b;联机用户接口即为终端&#xff08;所有输…

终端的启动流程

在Linux操作系统启动时&#xff0c;首先加载的进程就是init进程&#xff08;ID为1&#xff09;&#xff0c;其余进程都是init进程产生的&#xff08;fork&#xff0c;然后exec金蝉脱壳&#xff09;&#xff0c;因此系统中所有进程都可以看成是init进程的子孙进程。可以通过ps a…

进程组(作业)

&#xff08;1&#xff09;概念和特性 进程组&#xff0c;也称之为作业。BSD于1980年前后向Unix中增加的一个新特性。代表一个或多个进程的集合。每个进程都属于一个进程组。在waitpid函数和kill函数的参数中都曾使用到。操作系统设计的进程组的概念&#xff0c;是为了简化对多…

437. 路径总和 III

给定一个二叉树&#xff0c;它的每个结点都存放着一个整数值。 找出路径和等于给定数值的路径总数。 路径不需要从根节点开始&#xff0c;也不需要在叶子节点结束&#xff0c;但是路径方向必须是向下的&#xff08;只能从父节点到子节点&#xff09;。 二叉树不超过1000个节…

会话(session)

一组进程形成一个进程组&#xff0c;一组进程组形成一个会话&#xff0c;即一个会话中可以包括多个进程组。 &#xff08;1&#xff09;创建会话 创建一个会话需要注意以下6点注意事项&#xff1a;1.调用进程不能是进程组组长&#xff08;不能是父进程&#xff09;&#xff0…

508. 出现次数最多的子树元素和

给出二叉树的根&#xff0c;找出出现次数最多的子树元素和。一个结点的子树元素和定义为以该结点为根的二叉树上所有结点的元素之和&#xff08;包括结点本身&#xff09;。然后求出出现次数最多的子树元素和。如果有多个元素出现的次数相同&#xff0c;返回所有出现次数最多的…

1003 我要通过!(20)(20 分)

“答案正确”是自动判题系统给出的最令人欢喜的回复。本题属于PAT的“答案正确”大派送 —— 只要读入的字符串满足下列条件&#xff0c;系统就输出“答案正确”&#xff0c;否则输出“答案错误”。 得到“答案正确”的条件是&#xff1a; 1. 字符串中必须仅有P, A, T这三种字符…

网络终端

虚拟终端或串口终端的数目是有限的&#xff0c;虚拟终端&#xff08;字符控制终端&#xff09;一般就是/dev/tty1∼/dev/tty6六个&#xff0c;串口终端的数目也不超过串口的数目。然而网络终端或图形终端窗口的数目却是不受限制的&#xff0c;这是通过伪终端&#xff08;Pseudo…

线程的概念

线程&#xff08;LWP&#xff0c;light weight process&#xff09;是轻量级的进程&#xff0c;本质仍是进程&#xff08;在类unix环境下&#xff09;。进程有独立地址空间&#xff0c;拥有PCB&#xff1b;线程也有PCB&#xff0c;但没有独立的地址空间&#xff08;共享&#x…

1001. 害死人不偿命的(3n+1)猜想 (15)

卡拉兹(Callatz)猜想&#xff1a; 对任何一个自然数n&#xff0c;如果它是偶数&#xff0c;那么把它砍掉一半&#xff1b;如果它是奇数&#xff0c;那么把(3n1)砍掉一半。这样一直反复砍下去&#xff0c;最后一定在某一步得到n1。卡拉兹在1950年的世界数学家大会上公布了这个猜…

海量数据处理 (一)

现有海量日志数据保存在一个超级大的文件中&#xff0c;该文件无法直接读入内存&#xff0c;要求从中提取某天出访问百度次数最多的那个IP。 从这一天的日志数据中把访问百度的IP取出来&#xff0c;逐个写入到一个大文件中;注意到IP是32位的&#xff0c;最多有2^32个IP。同样可…

线程控制原语之pthread_self和pthread_create函数

注意&#xff1a;使用线程库函数用gcc编译时&#xff0c;要加参数&#xff1a;-lpthread&#xff08;libpthread.so&#xff09;&#xff0c;因为线程库函数属于第三方c库函数&#xff0c;不是标准库函数&#xff08;/lib、/usr/lib或者/usr/local/lib&#xff09;。 &#xf…

1005. 继续(3n+1)猜想 (25)

卡拉兹(Callatz)猜想已经在1001中给出了描述。在这个题目里&#xff0c;情况稍微有些复杂。 当我们验证卡拉兹猜想的时候&#xff0c;为了避免重复计算&#xff0c;可以记录下递推过程中遇到的每一个数。例如对n3进行验证的时候&#xff0c;我们需要计算3、5、8、4、2、1&#…

C指针深度解析

&#xff08;1&#xff09;指针的概念 指针是一种数据类型&#xff0c;而内存地址是这种数据类型具体的值&#xff08;注意区分两者的概念&#xff09;。先说一下什么是内存地址&#xff1a;假设CPU的寻址方式是以字节寻址的&#xff0c;即每一个字节对应一个地址编号&#xf…

1007. 素数对猜想

让我们定义 dn 为&#xff1a;dn pn1 - pn&#xff0c;其中 pi 是第i个素数。显然有 d11 且对于n>1有 dn 是偶数。“素数对猜想”认为“存在无穷多对相邻且差为2的素数”。 现给定任意正整数N (< 105)&#xff0c;请计算不超过N的满足猜想的素数对的个数。 输入格式&…

线程共享全局变量(.data和.bbs)

线程默认共享数据段、代码段等地址空间&#xff0c;常用的是全局变量。而进程不共享全局变量&#xff0c;只能借助mmap。 //代码示例 #include <string.h> #include <stdio.h> #include <pthread.h> #include <stdlib.h> #include <string.h> …

1008 数组元素循环右移问题 (20)

一个数组A中存有N&#xff08;N&gt0&#xff09;个整数&#xff0c;在不允许使用另外数组的前提下&#xff0c;将每个整数循环向右移M&#xff08;M>0&#xff09;个位置&#xff0c;即将A中的数据由&#xff08;A~0~ A~1~……A~N-1~&#xff09;变换为&#xff08;A~N-…