【Linux C | 网络编程】进程池退出的实现详解(五)

上一篇中讲解了在进程池文件传输的过程如何实现零拷贝,具体的方法包括使用mmap,sendfile,splice等等。

【Linux C | 网络编程】进程池零拷贝传输的实现详解(四)

这篇内容主要讲解进程池如何退出。

1.进程池的简单退出

进程池的简单退出要实现功能很简单,就是让父进程收到信号之后,再给每个子进程发送信号使其终止,这种实现方案只需要让父进程在一个目标信号(通常是10 信号 SIGUSR1 )的过程给目标子进程发送信号即可。
在实现的过程需要注意的是 signal 函数和 fork 函数之间调用顺序,因为父进程会修改默认递送行为,而子进程会执行默认行为,所以 fork 应该要在 signal 的前面调用。
processData_t *workerList;//需要改成全局变量
int workerNum;
void sigFunc(int signum){printf("signum = %d\n", signum);for(int i = 0; i < workerNum; ++i){kill(workerList[i].pid,SIGUSR1);}for(int i = 0; i < workerNum; ++i){wait(NULL);}puts("process pool is over!");exit(0);
}
int main(){
//..makeChild(workerList,workerNum);signal(SIGUSR1,sigFunc);//注意fork和signal的顺序
}

2.使用管道通知工作进程终止

采用信号就不可避免要使用全局变量,因为信号处理函数当中只能存储有限的信息,有没有办法避免全局的进程数量和进程数组呢? 一种解决方案就是采取“ 异步拉起同步 的策略:虽然还是需要创建一个管道全局变量,但是该管道只用于处理进程池退出,不涉及其他的进程属性。这个管道的读端需要使用IO多路复用机制管理起来,而当信号产生之后,主进程递送信号的时候会往管道中写入数据,此时可以依靠 epoll 的就绪事件,在事件处理中来完成退出的逻辑。

异步执行与同步执行的区别

  • 异步执行:异步操作是指当一个任务开始执行后,控制流可以继续执行下一个任务,而不需要等待当前任务完成。异步操作通常会通过回调函数、事件驱动机制或者Promise等方式来处理结果或通知任务的完成状态。在异步操作中,调用者通常不会立即等待操作的完成。

  • 同步执行:同步操作是指一个任务开始执行后,调用者会一直等待任务完成,然后才能继续执行下一个任务。同步操作的执行顺序是按照代码的顺序依次执行的,直到当前任务完成。

异步拉起同步的应用场景和解释

异步拉起同步通常用来描述这样一种情况:

  1. 异步任务的启动:首先,某个操作或任务以异步的方式启动,这意味着调用者可以在任务启动后继续执行其他操作,而不必等待任务完成。

  2. 同步任务的结束:随后,在异步任务执行完毕后,系统或程序需要在某个点上进行同步操作,即等待这个异步任务完成并获取其结果,然后才能继续执行依赖于该结果的下一步操作。

进程池退出中的异步拉起同步的具体流程

  1. 异步提交任务:主进程使用进程池异步地提交多个任务给子进程执行。这些任务可以是函数、方法或者其他可调用对象。

  2. 任务执行:进程池管理器负责分配任务给空闲的子进程,并在子进程完成任务后收集返回的结果。

  3. 等待任务完成:一旦主进程不再提交新任务到进程池,它需要等待所有已提交的任务都完成。这时候,主进程通常会发出一个信号或者事件,表示进程池应该开始进入退出状态。

  4. 异步转同步:进程池收到退出信号后,开始等待所有任务完成。此时,进程池中的子进程继续处理它们的任务,主进程则通过某种机制(例如等待事件或轮询任务状态)来异步地等待所有子进程的任务完成。

  5. 进程池退出:一旦所有任务完成,进程池中的子进程会被优雅地退出。主进程也可以在这个时候执行一些清理工作,例如关闭文件、释放资源等。

简单点总结就是:

信号 + 匿名管道  =》 异步拉起同步

1. SIGUSR1  父进程获取信号后,在信号处理函数中,通过匿名管道进入epoll

2. 在epoll循环中,有两种方式让子进程退出     

A. 在父进程中,直接调用kill函数,给子进程发送SIGUSR1信号(粗暴)     

B. 在父进程中,通过sendFd函数,通知子进程退出(温和)在子进程中,通过recvFd函数,获取到退出标志位,然后退出,可以确保每一个任务都能够执行完毕。

粗暴的退出代码:

#include "process_pool.h"//退出使用的匿名管道
int exitPipe[2];void sighandler(int signum)
{printf("signum %d is coming.\n", signum);int one = 1;//父进程收到信号,往管道写端写入数据write(exitPipe[1], &one, sizeof(one));
}int main(int argc, char ** argv)
{//ip port processnumARGS_CHECK(argc, 4);int processNum = atoi(argv[3]);process_data * pProcess = calloc(processNum, sizeof(process_data));//让父子进程都忽略掉SIGPIPE信号//signal(SIGPIPE, SIG_IGN);//创建N个子进程makeChild(pProcess, processNum);//makechild函数之后,都是父进程的操作//在父进程中注册信号处理函数signal(SIGUSR1, sighandler);//只在父进程中创建退出的管道pipe(exitPipe);//创建监听的服务器int listenfd = tcpInit(argv[1], atoi(argv[2]));//创建epoll的实例int epfd = epoll_create1(0);ERROR_CHECK(epfd, -1, "epfd");//epoll监听ListenfdepollAddReadEvent(epfd, listenfd);//epoll监听进程池退出的管道读端exitPipe[0],epollAddReadEvent(epfd, exitPipe[0]);///epoll监听父子进程间通信的管道for(int i = 0; i < processNum; ++i) {epollAddReadEvent(epfd, pProcess[i].pipefd);}//定义保存就绪的文件描述符的数组struct epoll_event eventArr[10] = {0};int nready = 0;while(1){nready = epoll_wait(epfd, eventArr, sizeof(eventArr), -1);for(int i = 0; i < nready; ++i) {int fd = eventArr[i].data.fd; //新客户端到来if(fd == listenfd) {struct sockaddr_in clientaddr;socklen_t len = sizeof(clientaddr);int peerfd = accept(listenfd, (struct sockaddr*)&clientaddr, &len);ERROR_CHECK(peerfd, -1, "accept");printf("client %s:%d connected.\n",inet_ntoa(clientaddr.sin_addr), ntohs(clientaddr.sin_port));//将peerfd发送给一个空闲的子进程for(int j = 0; j < processNum; ++j) {if(pProcess[j].status == FREE) {sendFd(pProcess[j].pipefd, peerfd);pProcess[j].status = BUSY;break;}}//如果要断开与客户端的连接,这里还得执行一次close(peerfd);} else if(fd == exitPipe[0])  {     //匿名管道读端有数据,说明父进程收到退出的信号,循环给子进程发出退出信号int howmany = 0;read(fd, &howmany, sizeof(howmany));//处理进程池退出的情况//第一种方式: 父进程给子进程发送SIGUSR1信号for(int j = 0; j < processNum; ++j) {kill(pProcess[j].pid, SIGUSR1);}for(int j = 0; j < processNum; ++j) {wait(NULL);}goto end;} else {//管道发生了事件: 子进程已经执行完任务了int howmany = 0;read(fd, &howmany, sizeof(howmany));for(int j = 0; j < processNum; ++j) {if(pProcess[j].pipefd == fd) {pProcess[j].status = FREE;printf("child %d is not busy.\n", pProcess[j].pid);break;}}}}}
end:printf("exit process pool.\n");free(pProcess);close(exitPipe[0]);close(exitPipe[1]);close(listenfd);close(epfd);return 0;
}

3.优雅退出

上述的退出机制存在一个问题,就是即使工作进程正在传输文件中,父进程也会通过信号将其终止。如何实现进程池在退出的时候,子进程要完成传输文件的工作之后才能退出呢?
一种典型的方案是使用 sigprocmask 在文件传输的过程中设置信号屏蔽字,这样可以实现上述的机制。
另一种方案就是调整 sendFd 的设计,每个工作进程在传输完文件之后总是循环地继续下一个事件,而在每个事件处理的开始,工作进程总是会调用 recvFd 来使自己处于阻塞状态直到有事件到达。我们可以对进程池的终止作一些调整:用户发送信号给父进程表明将要退出进程池;随后父进程通过 sendFd给所有的工作进程发送终止的信息,工作进程在完成了一次工作任务了之后就会 recvFd 收到进程池终止的信息,然后工作进程就可以主动退出;随着所有的工作进程终止,父进程亦随后终止,整个进程池就终止了。

int sendFd(int pipeFd, int fdToSend, int exitFlag){struct msghdr hdr;bzero(&hdr,sizeof(struct msghdr));struct iovec iov[1];iov[0].iov_base = &exitFlag;iov[0].iov_len = sizeof(int);hdr.msg_iov = iov;hdr.msg_iovlen = 1;//...
}
int recvFd(int pipeFd, int *pFd, int *exitFlag){struct msghdr hdr;bzero(&hdr,sizeof(struct msghdr));struct iovec iov[1];iov[0].iov_base = exitFlag;iov[0].iov_len = sizeof(int);hdr.msg_iov = iov;hdr.msg_iovlen = 1;//.....
}

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

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

相关文章

Java并发编程(上)

并发&#xff1a;多个线程&#xff08;进程&#xff09;竞争一个资源 并行&#xff1a;多个线程&#xff08;进程&#xff09;同时运行不同资源 线程和进程的关系简单地说&#xff0c;进程是一个容器&#xff0c;一个进程中可以容纳若干个线程&#xff0c;一个进程里面&#…

微信小程序入门

创建一个入门程序 这是index.vxml代码 <!--index.wxml--> <navigation-bar title"Weixin" back"{{false}}" color"black" background"#FFF"></navigation-bar> <view class"container" ><view&…

苹果CMS:资源采集站如何设置定时采集详细教程讲解

我们搭建好站点之后&#xff0c;会自定义一些采集&#xff0c;但是需要每天去手动执行&#xff0c;有时候甚至会忘记&#xff0c;那我们如何处理呢&#xff1f;今天我们就来介绍一下如何设置定时器。 如果按照官方例子来设置定时器会遇到一个问题就是采集的资源未绑定类型&…

WAF+API安全代表厂商|瑞数信息入选IDC报告《生成式AI推动下的中国网络安全硬件市场现状及技术发展趋势》

近日&#xff0c;全球领先的权威资讯机构IDC正式发布《IDC Market Presentation&#xff1a;生成式AI推动下的中国网络安全硬件市场现状及技术发展趋势&#xff0c;2024》报告。报告中IDC 评估了众多厂商的安全硬件产品能力&#xff0c;并给出了产品对应的推荐厂商供最终用户参…

04 | 深入浅出索引(上)

此系列文章为极客时间课程《MySQL 实战 45 讲》的学习笔记&#xff01; 索引的常见模型 可以提供查询效率的数据结构有很多&#xff0c;常见的有三种&#xff1a;哈希表、有序数组、搜索数。 哈希表是一种以 key-value 形式存储的数据结构。输入一个 key&#xff0c;通过固定…

强烈推荐java人,2024年大厂面试背这份(八股文+场景题结合)!很管用!

2024 年的行情&#xff0c;和 3~4 年前不同&#xff0c;通过海量简历投递和海量面试找工作的时代已经过去了。 在如今面试机会较少&#xff0c;并且面试难度较大的情况下。 充分做好面试的准备才是快速通过面试最有效的方法&#xff01; 切忌把真实面试当靶场&#xff0c;最…

信息学奥赛初赛天天练-48-CSP-J2020完善程序2-变量交换、冒泡排序、贪心算法、最小区间覆盖

PDF文档公众号回复关键字:20240728 2020 CSP-J 完善程序2 1 完善程序 (单选题 &#xff0c;每小题3分&#xff0c;共30分) 最小区间覆盖 给出 n 个区间&#xff0c;第 i 个区间的左右端点是 [ai,bi]。现在要在这些区间中选出若干个&#xff0c;使得区间 [0, m] 被所选区间的…

前端框架 element-plus 发布 2.7.8

更新日志 功能 组件 [级联选择器 (cascader)] 添加持久化属性以提升性能 (#17526 by 0song)[日期选择器 (date-picker)] 类型添加月份参数 (#17342 by Panzer-Jack)[级联选择器 (cascader)] 添加标签效果属性 (#17443 by ntnyq)[加载 (loading)] 补充加载属性 (#17174 by zhixi…

第九讲 后端1

后端&#xff08;Backend&#xff09; 用带噪声的数据估计内在状态&#xff08;Estimated the inner state from noisy data&#xff09;——状态估计问题渐进式&#xff08;Incremental&#xff09;&#xff1a;保持当前状态的估计&#xff0c;在假如新信息时&#xff0c;更新…

【算法专题】双指针算法之18. 四数之和(力扣)

欢迎来到 CILMY23的博客 &#x1f3c6;本篇主题为&#xff1a;双指针算法之18. 四数之和&#xff08;力扣&#xff09; &#x1f3c6;个人主页&#xff1a;CILMY23-CSDN博客 &#x1f3c6;系列专栏&#xff1a;Python | C | C语言 | 数据结构与算法 | 贪心算法 | Linux | 算…

ProxmoxPVE虚拟化平台--U盘挂载、硬盘直通

界面说明 ### 网络设置 ISO镜像文件 虚拟机中使用到的磁盘 挂载USB设备 这个操作比较简单&#xff0c;不涉及命令 选中需要到的虚拟机&#xff0c;然后选择&#xff1a; 添加->USB设置选择使用USB端口&#xff1a;选择对应的U盘即可 硬盘直通 通常情况下我们需要将原有…

【Linux 16】进程间通信的方式 - 共享内存

文章目录 &#x1f308; 一、共享内存概述⭐ 1. 什么是共享内存⭐ 2. 如何实现共享内存⭐ 3. 操作系统允许存在多个共享内存⭐ 4. 操作系统如何管理共享内存⭐ 5. 获取共享内存的唯一标识符 key⭐ 6. 为什么要由用户提供 key &#x1f308; 二、查看共享内存⭐ 1. 使用 ipcs -m…

TCP 协议的 time_wait 超时时间

优质博文&#xff1a;IT-BLOG-CN 灵感来源 Time_Wait 产生的时机 TCP四次挥手的流程 如上所知&#xff1a;客户端在收到服务端第三次FIN挥手后&#xff0c;就会进入TIME_WAIT状态&#xff0c;开启时长为2MSL的定时器。 【1】MSL是Maximum Segment Lifetime报文最大生存时间…

root 用户和权限

目录 1. 超级管理员 root 2. 切换用户 Switch User 2.1 普通用户切换到 root 用户 2.2 root 用户切换到普通用户 3. sudo 命令 3.1 配置认证 无论是 Windows&#xff0c;MacOS&#xff0c;Linux 均采用多用户的管理模式管理权限&#xff1b; 1. 超级管理员 root 在 Li…

2年社招冲击字节,一天三面斩获offer

在工作满两年的时间选择了求变&#xff0c;带着运气和实力以社招身份重新看今天的互联网环境&#xff0c;从结果看还是复合预期的。 整个面试的流程还挺快的。周中让招聘专员给投递了简历。问什么时候面试&#xff0c;申请了一个周日&#xff0c;直接安排三面。下周周中就开启…

C#中的wpf基础

在WPF中&#xff0c;Grid 是一种非常强大的布局控件&#xff0c;用于创建网格布局。它允许你将界面划分为行和列&#xff0c;并将控件放置在这些行和列中。 以下是一些关键点和示例&#xff0c;帮助你理解 WPF 中的 Grid&#xff1a; 基本属性 RowDefinitions&#xff1a;定义…

[MIT6.5840]MapReduce

MapReduce Lab 地址 https://pdos.csail.mit.edu/6.824/labs/lab-mr.html 论文地址 https://static.googleusercontent.com/media/research.google.com/zh-CN//archive/mapreduce-osdi04.pdf 工作原理 简单来讲&#xff0c;MapReduce是一种分布式框架&#xff0c;可以用来处理…

windows 安装docker桌面版

下载 下载两个&#xff1a; git桌面版 docker desktop 启动docker 执行安装文件&#xff0c;启动 更新wsl2 假如报错&#xff0c;会提示失败原因。 win10会提示跳转到&#xff1a; https://learn.microsoft.com/zh-cn/windows/wsl/install-manual#step-4—download-the-l…

从0到1,AI我来了- (4)AI图片识别的理论知识-II

上篇文章&#xff0c;我们理解了我们程序的神经网络设计&#xff0c;这篇我们继续&#xff0c;把训练迭代过程分析一下&#xff0c;完成这两篇文章&#xff0c;下面问题&#xff0c;应该能回答了。 一张图片&#xff0c;如何被计算机读懂&#xff1f;pytorch 封装的网络&#…

DP 整数拆分不同的二叉搜索树 DAY21

整数拆分&#xff1f; 给定一个正整数 n &#xff0c;将其拆分为 k 个 正整数 的和&#xff08; k > 2 &#xff09;&#xff0c;并使这些整数的乘积最大化。 返回 你可以获得的最大乘积。 示例 1: 输入: n 2 输出: 1 解释: 2 1 1, 1 1 1。示例 2: 输入: n 10 输…