为什么 while((pid = waitpid(-1, &stat, WNOHANG)) > 0)能处理所有子进程

在unp中有一节代码是这样的:

void sig_chld(int signo)
{
       pid_t   pid;
       int     stat;
       
       while((pid = waitpid(-1, &stat, WNOHANG)) > 0){
               printf("child %d terminated\n", pid);
       }
        return;
}

见unp第110页。

是什么?

它是一个SIGCHLD信号处理程序。当父进程的一个子进程结束时,会给父进程发送一个SIGCHLD信号。

核心在哪儿?

当父进程的5个子进程同时向父进程发送SIGCHILD时会发生什么?

父进程只会调用一次信号处理程序,并丢弃其余四个信号。虽然只接受了一个SIGCHLD信号,但是

while((pid = waitpid(-1, &stat, WNOHANG)) > 0)

这条语句会处理所有的僵尸子进程。这就意味着,即使丢失了一部分SIGCHLD信号,父进程也能够回收所有的子进程。

为什么?

事实上,waitpid与SIGCHLD信号的关联并不大。waitpid的函数构造是,回收第一个僵尸子进程。而加上while之后就能够回收所有在while运行期间内结束的子进程。与SIGCHLD关系不大。

 

#include <stdio.h> #include <stdlib.h> #include <stdbool.h> #include <string.h> #include <sys/wait.h> #include <sys/types.h> #include <sys/stat.h> #include <errno.h> #include <unistd.h> // for execlp #include <mqueue.h> // for mq #include "settings.h" #include "messages.h" char client2dealer_name[30]; int groupnumber = 16; pid_t processID; mqd_t Req_queue_X; MQ_REQUEST_MESSAGE req; int main (int argc, char * argv[]) { struct mq_attr attr; attr.mq_maxmsg = MQ_MAX_MESSAGES; attr.mq_msgsize = sizeof (MQ_REQUEST_MESSAGE); sprintf (client2dealer_name, "/Req_queue_%d", groupnumber); mq_unlink(client2dealer_name); Req_queue_X = mq_open (client2dealer_name, O_RDONLY | O_CREAT | O_EXCL, 0600, &attr); if(Req_queue_X == -1) { perror("Channel creation failed!"); } getattr(Req_queue_X); processID = fork(); if (processID < 0) { perror("fork() failed"); exit (1); } else if (processID == 0) { // child-stuff execlp("./client", "client", client2dealer_name, NULL); perror("execlp failed"); exit (0); } while (1) { int status; pid_t result = waitpid(processID, &status, WNOHANG); // Non-blocking check if (result > 0) { printf("Client terminated\n"); break; } ssize_t size = mq_receive(Req_queue_X, (char*)&req, sizeof(MQ_REQUEST_MESSAGE), 0); if (size == -1) { perror("Receiving failed \n"); } else if ( size == sizeof(MQ_REQUEST_MESSAGE)) { if (req.serv_type == 1 || req.serv_type == 2) { printf("Received message with id %d, type %d and value %d\n", req.req_id, req.serv_type, req.input); } else if (req.serv_type == 0) { printf("...\n"); } } else { perror("mq_receive failed partially \n"); } sleep(1); printf("I'm still alive \n"); } printf("loop ends \n"); waitpid(processID, NULL, 0); mq_close(Req_queue_X); mq_unlink(client2dealer_name); } 帮我分析下问题在哪里
03-14
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值