进程池设计

进程池设计

c++d281926eda>进程池设计

代码目的头文件代码语言:c++复制

#include#include#include#include#include#include#include#include

对子进程操作建立子进程对象并把子进程对象放进数组里代码语言:c++复制

//创建子进程对象class  SubEp//endpoint---子进程对象{public:SubEp(pid_t subid,int writefd)//第一个参数是子进程的pid,第二个参数是该子进程读端对于父进程的写端fd:_subid(subid),_writefd(writefd){   char namebuffer[1024];   //第一个参数是表示第几号子进程,第二个参数是子进程的pid,第三个参数是该子进程读端对于的父进程的写端   snprintf(namebuffer,sizeof namebuffer,"process: %d [pid(%d) - fd(%d)]",num++,_subid,_writefd);   _name=namebuffer;}public:static int num;string _name;pid_t _subid;int _writefd;//该子进程与父进程匿名管道对于的父进程的写端fd};int SubEp::num=0;

子进程对象需要传递两个参数来初始化成员变量_subid和 _writefd。一是子进程的pid二是该子进程读端对应父进程写端的文件描述符fd成员变量num表示是第几个创建出来的子进程,第一个创建出来的子进程为0,使用后++后续子进程的num依次是1,2等等。因此num不能由于出了SubEp对象作用域后被销毁,所以定义为static,变量num生命周期取决于SubEp类的生命周期成员变量 _name用namebuffer初始化,用来标识该子进程的其他成员变量建立子进程需要执行的任务表代码语言:c++复制

//创建父进程给子进程派发的任务列表typedef void(*func_t)();//函数指针类型,函数返回值为voidvoid downloadTask()//模拟下载任务{    cout<<getpid()<<": 下载任务n"<<endl;    sleep(1);}void fflushTask()//模拟刷新任务{    cout<<getpid()<<": 刷新任务n"<<endl;    sleep(1);}void subscribeTask()//模拟订阅任务{    cout<<getpid()<<": 订阅任务n"<<endl;    sleep(1);}//把上面的三种任务load到列表中即让存放函数指针的vector的各个指针能够指向上面的函数,为了后面方便调用void loadTaskFunc(vector*out){    assert(out);//vector创建成功    out->push_back(downloadTask);    out->push_back(fflushTask);    out->push_back(subscribeTask);}

子进程需要执行的任务都是函数对象,建立一个对象是函数指针的数组out,通过loadTaskFunc函数把任务函数尾插到数组out里,然后通过输出型参数返回。创建子进程和父进程通信的管道,并且让子进程阻塞读取代码语言:c++复制

void CreateSubProcesses( vector*subs,vector& funcMap){    vector deleteFd;//创建子进程并且创建好父进程与各个子进程通信的管道int fds[2];for(size_t i=0;i<PROCESS_NUM;i++)//创建子进程{        int n=pipe(fds);//建立父子间进程的匿名管道--建立成功返回0,建立失败返回-1        assert(n==0);//判断管道是否建立成功        (void)n;        pid_t id=fork();//创建子进程        if(id==0)        {             for(size_t i=0;i=0 && taskcodepush_back(sub);        deleteFd.push_back(fds[1]);//记录当前的写端供下个子进程用}

在函数CreateSubProcesses内,先建立父进程相连的匿名管道,然后创建子进程,子进程也拷贝了一份父进程的文件描述符表,能通过文件描述符连接到匿名管道,因此父子进程通信的管道建立完成。在父进程语句中,需要注意的是,通过传参数子进程的pid和此时子进程读端对于的父进程的写端fd给SubEP类构建子进程对象,并且将对象放进数组subs里。在子进程的语句中,通过receiveTask函数获取任务码代码语言:c++复制

int receiveTask(int readfd){    int retcode=0;//返回任务码    ssize_t s= read(readfd,&retcode,sizeof(retcode));//从读端读出来的任务码放到retcode里    cout<<"process has read the TaskCode: "<<retcode<<endl;    if(s==sizeof(int)) return retcode;    else if(s<=0)return -1;    else return 0;}

让子进程在receiveTask函数中阻塞读取管道里的数据。前提已知父进程往匿名管道写入整数数据,数据范围为0,任务个数-1即任务数组对应的下标范围,子进程把读取到的数据存到变量retcode里,然后判断retcode是否是整数数据大小,如果是就返回数据给上层CreateSubProcesses函数,如果不是就返回-1。当变量taskcode接收到receiveTask函数返回的任务码时,如果任务码符合范围0,任务个数-1即父子进程按照我们的意愿通信正常,然后子进程拿着任务码调用funcMap数组执行任务;但如果接收的返回值是-1,则是父子进程间通信不正常,直接退出判断语句。

这里提到的子进程操作主要是子进程阻塞读取父进程写入的数据,还有子进程拿到数据执行任务。

对父进程操作代码语言:c++复制

void loadBalanceContrl(const vector& subs,const vector &funcMap,int comcode){    int processnum=subs.size();//子进程的个数    int tasknum=funcMap.size();//任务的个数    bool numoftime=(comcode==0?true:false);//若命令码是0则一直运行,若命令码为正数x,则允许x次后退出    while(true)    {   //rand()为伪随机数   //1.找到哪一个子进程   int subIndex=rand()%processnum;    //2.找到哪一个执行哪一个任务   int taskIndex=rand()%tasknum;    //3.任务发送给选择的进程     sendTask(subs[subIndex],taskIndex);//第一个参数传第几个子进程,第二个参数传第几个任务     sleep(1);  if(!numoftime)  {    comcode--;    if(comcode==0)    break;  }    }    //走到这里则是父进程给子进程通信完了,需要逐个关闭子进程读端对于的写端--倒退关解决bug    for(size_t i=0;i<subs.size();i++)    {      close(subs[i]._writefd);      cout<<"close process: [ "<<i<<" ]'s writeeop"<<endl;    }}

loadBalanceContrl函数需要main函数传入子进程数组subs,任务数组funcMap和命令码comcode。comcode用来指定父进程发送多少次数据给子进程即子进程需要执行多少次任务numoftime用来鉴别父进程需要写入多少次数据,当comcode为0时则numoftime为真,则父进程死循环往匿名管道里写数据;若命令码为正数x为非0,则numoftime为假,则父进程往匿名管道里写x次数据。通过sendTask函数让父进程选择指定的子进程,写入指定的任务码到匿名管道中代码语言:c++复制

void sendTask(const SubEp& process, int tasknum){    cout<<"send Task num: "<<tasknum<<" to the process: "<<process._name<<endl;//打印日志:任务几发送给几号子进程    ssize_t n=write(process._writefd,&tasknum,sizeof(tasknum));//该子进程读端对于的写端往管道里写入任务几-4个字节的数据    assert(n==sizeof(int));//判断写入的数据是否是4个字节    (void)n;}

父子间进程通信完之后,按照子进程创建时间从先往后依次关闭子进程读端对应的父进程的写端。回收子进程代码语言:c++复制

void waitProcess(const vector& processes){    for(size_t i=0;i<processes.size();i++)    {        waitpid(processes[i]._subid,nullptr,0);        cout<<"wait success for process: "<<processes[i]._subid<<endl;    }}

按照子进程创建时间从先往后依次回收子进程。整体代码代码语言:c++复制

#include#include#include#include#include#include#include#includeusing namespace std;#define PROCESS_NUM 3#define MakeSeed() srand((unsigned long)time(nullptr)^getpid()^rand()%1234)//建立伪随机数种子//创建父进程给子进程派发的任务列表typedef void(*func_t)();//函数指针类型,函数返回值为voidvoid downloadTask()//模拟下载任务{    cout<<getpid()<<": 下载任务n"<<endl;    sleep(1);}void fflushTask()//模拟刷新任务{    cout<<getpid()<<": 刷新任务n"<<endl;    sleep(1);}void subscribeTask()//模拟订阅任务{    cout<<getpid()<<": 订阅任务n"<<endl;    sleep(1);}//把上面的三种任务load到列表中即让存放函数指针的vector的各个指针能够指向上面的函数,为了后面方便调用void loadTaskFunc(vector*out){    assert(out);//vector创建成功    out->push_back(downloadTask);    out->push_back(fflushTask);    out->push_back(subscribeTask);}//创建子进程对象class  SubEp//endpoint---子进程对象{public:SubEp(pid_t subid,int writefd)//第一个参数是子进程的pid,第二个参数是该子进程读端对于父进程的写端fd:_subid(subid),_writefd(writefd){   char namebuffer[1024];   //第一个参数是表示第几号子进程,第二个参数是子进程的pid,第三个参数是该子进程读端对于的父进程的写端   snprintf(namebuffer,sizeof namebuffer,"process: %d [pid(%d) - fd(%d)]",num++,_subid,_writefd);   _name=namebuffer;}public:static int num;string _name;pid_t _subid;int _writefd;//该子进程与父进程匿名管道对于的父进程的写端fd};int SubEp::num=0;int receiveTask(int readfd){    int retcode=0;//返回任务码    ssize_t s= read(readfd,&retcode,sizeof(retcode));//从读端读出来的任务码放到retcode里    cout<<"process has read the TaskCode: "<<retcode<<endl;    if(s==sizeof(int)) return retcode;    else if(s<=0)return -1;    else return 0;}void CreateSubProcesses( vector*subs,vector& funcMap){    vector deleteFd;//创建子进程并且创建好父进程与各个子进程通信的管道int fds[2];for(size_t i=0;i<PROCESS_NUM;i++)//创建子进程{        int n=pipe(fds);//建立父子间进程的匿名管道--建立成功返回0,建立失败返回-1        assert(n==0);//判断管道是否建立成功        (void)n;        pid_t id=fork();//创建子进程        if(id==0)//子进程进入判断语句        {            for(size_t i=0;i=0 && taskcodepush_back(sub);       deleteFd.push_back(fds[1]);//记录当前的写端供下个子进程用}}void sendTask(const SubEp& process, int tasknum){    cout<<"send Task num: "<<tasknum<<" to the process: "<<process._name<<endl;//打印日志:任务几发送给几号子进程    ssize_t n=write(process._writefd,&tasknum,sizeof(tasknum));//该子进程读端对于的写端往管道里写入任务几-4个字节的数据    assert(n==sizeof(int));//判断写入的数据是否是4个字节    (void)n;}void loadBalanceContrl(const vector& subs,const vector &funcMap,int comcode){    int processnum=subs.size();//子进程的个数    int tasknum=funcMap.size();//任务的个数    bool numoftime=(comcode==0?true:false);//若命令码是0则一直运行,若命令码为正数x,则允许x次后退出    while(true)    {        //rand()为伪随机数   //1.找到哪一个子进程   int subIndex=rand()%processnum;    //2.找到哪一个执行哪一个任务   int taskIndex=rand()%tasknum;    //3.任务发送给选择的进程     sendTask(subs[subIndex],taskIndex);//第一个参数传第几个子进程,第二个参数传第几个任务     sleep(1);  if(!numoftime)  {    comcode--;    if(comcode==0)    break;  }    }    //走到这里则是父进程给子进程通信完了,需要逐个关闭子进程读端对于的父进程写端    for(size_t i=0;i<subs.size();i++)    {      close(subs[i]._writefd);      cout<<"close process: [ "<<i<<" ]'s writeeop"<<endl;    //    waitpid(subs[i]._subid,nullptr,0);    //     cout<<"wait success for process: "<<subs[i]._subid<<endl;    }}void waitProcess(const vector& processes){    for(size_t i=0;i<processes.size();i++)    {        waitpid(processes[i]._subid,nullptr,0);        cout<<"wait success for process: "<<processes[i]._subid<<endl;    }}int main(){    MakeSeed();//建立伪随机数种子vector subs;//创建子进程对象并将子进程对象放进数组里vector funcMap;//建立一个任务表:父进程写入管道,子进程在管道读取,读取到的数据引导子进程去完成一些任务loadTaskFunc(&funcMap);//1.创建子进程并且创建好父进程与各个子进程通信的管道,并且让子进程阻塞等待父进程写入CreateSubProcesses(&subs,funcMap);//2.对父进程操作//父进程给子进程发送命令码,为0则一直运行,为正数x则运行x次后退出int Runcount=0;cout<>Runcount;cout<<endl;//这个函数负责让父进程给子进程发送命令码,让子进程去执行任务,要求子进程做到负载均衡     loadBalanceContrl(subs,funcMap,Runcount);//第一个参数是子进程列表,第二个参数任务列表,第三个参数是父进程给子进程发送的命令码 //3.回收子进程waitProcess(subs);    return 0;}

子进程具有读端未关闭的bug

进程池设计image-20230522164129612
进程池设计image-20230522164147020
进程池设计image-20230522164456853

通过上面的图例可以看到,2号子进程有一个写端与1号子进程的读端通信着。

得出结论:当父进程创建多个子进程,并且父进程作为写端而多个子进程作为读端从而进行进程间通信时,需要单独把子进程的所有写端都关闭。

这里提供两种方法关闭子进程的所有写端

方案一:在父进程创建子进程前构建一个vector对象,父进程创建子进程后,把父进程的写端放进vector里。等到父进程创建完下一个子进程时,vector里的写端即是当前子进程读端对应的上一个子进程的写端(有可能不只是一个写端),再把vector里的所有写端关闭即可。

进程池设计image-20230522202611512
进程池设计image-20230522203254968

这次实现就是用的这个方案,其实不用也可以,因为当父进程往匿名管道里写完数据时,先把父进程对应各个子进程的写端全部关闭,然后再将全部子进程进行回收,这种顺序不会出现bug;但如果是按照创建子进程时间从旧往新关掉一个父进程的写端,然后立刻等待回收一个相应的子进程的话,会导致出现该子进程读端还有其他子进程的写端通信着,该子进程读端没有读到0导致子进程没有正常退出,那么父进程也就回收不到子进程。方案二:在父进程关闭写端,需要所有子进程关闭读端时,依次按照创建的时间从新往旧(从后往前)关闭父进程的写端。由于最后创建的子进程的读端只对应父进程的写端,那么父进程关闭写端时,最后一个子进程的读端读到0正常关闭读端,那么该子进程的文件描述符表也会被关闭,进而该子进程正常退出,从而该子进程连接着前一个子进程的写端也会被关闭;那么轮到下一个子进程时,该子进程的读端也只会对应父进程的写端,父进程关闭写端,子进程读端读到0正常关闭读端,子进程正常退出。

以上就是进程池设计的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/72356.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月13日 15:48:49
下一篇 2025年11月13日 16:09:10

相关推荐

  • 您不需要 CSS 预处理器

    原生 css 在最近几个月/几年里取得了长足的进步。在这篇文章中,我将回顾人们使用 sass、less 和 stylus 等 css 预处理器的主要原因,并向您展示如何使用原生 css 完成这些相同的事情。 分隔文件 分离文件是人们使用预处理器的主要原因之一。尽管您已经能够将另一个文件导入到 css…

    2025年12月24日
    000
  • React 嵌套组件中,CSS 样式会互相影响吗?

    react 嵌套组件 css 穿透影响 在 react 中,嵌套组件的 css 样式是否会相互影响,取决于采用的 css 解决方案。 传统 css 如果使用传统的 css,在嵌套组件中定义的样式可能会穿透影响到父组件。例如,在给出的代码中: 立即学习“前端免费学习笔记(深入)”; component…

    2025年12月24日
    000
  • React 嵌套组件中父组件 CSS 修饰会影响子组件样式吗?

    对嵌套组件的 CSS 修饰是否影响子组件样式 提问: 在 React 中,如果对嵌套组件 ComponentA 配置 CSS 修饰,是否会影响到其子组件 ComponentB 的样式?ComponentA 是由 HTML 元素(如 div)组成的。 回答: 立即学习“前端免费学习笔记(深入)”; 在…

    2025年12月24日
    000
  • 构建模拟:从头开始的实时交易模拟器

    简介 嘿,开发社区!我很高兴分享我的业余项目 Simul8or – 一个实时日间交易模拟器,旨在为用户提供一个无风险的环境来练习交易策略。该项目 100% 构建在 ASP.NET WebForms、C#、JavaScript、CSS 和 SQL Server 技术堆栈上,没有外部库或框架。从头开始构…

    2025年12月24日
    300
  • 如何在 VS Code 中解决折叠代码复制问题?

    解决 VS Code 折叠代码复制问题 在 VS Code 中使用折叠功能可以帮助组织长代码,但使用复制功能时,可能会遇到只复制可见部分的问题。以下是如何解决此问题: 当代码被折叠时,可以使用以下简单操作复制整个折叠代码: 按下 Ctrl + C (Windows/Linux) 或 Cmd + C …

    2025年12月24日
    000
  • 在 React 项目中实现 CSS 模块

    react 中的 css 模块是一种通过自动生成唯一的类名来确定 css 范围的方法。这可以防止大型应用程序中的类名冲突并允许模块化样式。以下是在 react 项目中使用 css 模块的方法: 1. 设置 默认情况下,react 支持 css 模块。你只需要用扩展名 .module.css 命名你的…

    2025年12月24日
    000
  • 姜戈顺风

    本教程演示如何在新项目中从头开始配置 django 和 tailwindcss。 django 设置 创建一个名为 .venv 的新虚拟环境。 # windows$ python -m venv .venv$ .venvscriptsactivate.ps1(.venv) $# macos/linu…

    2025年12月24日
    000
  • 花 $o 学习这些编程语言或免费

    → Python → JavaScript → Java → C# → 红宝石 → 斯威夫特 → 科特林 → C++ → PHP → 出发 → R → 打字稿 []https://x.com/e_opore/status/1811567830594388315?t=_j4nncuiy2wfbm7ic…

    2025年12月24日
    000
  • action在css中的用法

    CSS 中 action 关键字用于定义鼠标悬停或激活元素时的行为,语法:element:action { style-property: value; }。它可以应用于 :hover 和 :active 伪类,用于创建交互效果,如更改元素外观、显示隐藏元素或启动动画。 action 在 CSS 中…

    2025年12月24日
    000
  • css规则的类型有哪些

    CSS 规则包括:通用规则:选择所有元素类型选择器:根据元素类型选择元素类选择器:根据元素的 class 属性选择元素ID 选择器:根据元素的 id 属性选择元素(唯一)后代选择器:选择特定父元素内的元素子选择器:选择作为特定父元素的直接子元素的元素伪类:基于元素的状态或特性选择元素伪元素:创建元素…

    2025年12月24日
    000
  • css和c的区别是什么

    区别是:1、C语言是一门面向过程、抽象化的通用程序设计语言、计算机编程语言,广泛应用于底层开发;2、CSS是一种用来表现HTML或XML等文件样式的计算机语言,可以做到网页和内容进行分离的一种样式语言。 本教程操作环境:windows7系统、CSS3&&HTML5版、Dell G3电…

    2025年12月24日
    000
  • 如何查看编写的html_查看自己编写的HTML文件效果【效果】

    要查看HTML文件的浏览器渲染效果,需确保文件以.html为扩展名保存、用浏览器直接打开、利用开发者工具调试、必要时启用本地HTTP服务器、或使用编辑器实时预览插件。 如果您编写了HTML代码,但无法直观看到其在浏览器中的实际渲染效果,则可能是由于文件未正确保存、未使用浏览器打开或文件扩展名设置错误…

    2025年12月23日
    400
  • HTML5怎么制作广告_HTML5用动画与交互制横幅或弹窗广告吸引点击【制作】

    可利用HTML5结合CSS3动画、Canvas、Web Animations API、Intersection Observer和video标签制作互动广告:一用@keyframes实现横幅入场动画;二用Canvas绘制并响应悬停;三用Web Animations API控制弹窗时序;四用Inter…

    2025年12月23日
    000
  • html5怎么引用js_HTML5用外链或内嵌JS代码引用脚本【引用】

    HTML5中执行JavaScript需通过外链或内嵌方式引入:一、外链用,支持defer/async;二、内嵌将代码写入间,推荐置于body底部;三、type属性默认可省略;四、模块化使用type=”module”支持ES6 import/export。 <img sr…

    好文分享 2025年12月23日
    000
  • html5怎么设置黑体_html5用CSS font-family设黑体或font-weight加粗【设置】

    在HTML5中实现黑体及加粗需用CSS的font-family和font-weight:一、font-family按优先级列“SimHei”,“Microsoft YaHei”,“Heiti SC”,sans-serif;二、font-weight用700或bold;三、组合声明并注意继承;四、可用…

    2025年12月23日
    000
  • navigator怎么用html5_HTML5用navigator对象查浏览器信息如语言【对象】

    可通过navigator对象获取浏览器语言、设备类型、平台信息、地理定位和媒体设备支持:navigator.language/languages返回语言代码;userAgent判断移动设备;platform返回操作系统;geolocation检测定位支持;mediaDevices检查媒体访问能力。 …

    2025年12月23日
    000
  • html5框架怎么设置_HTML5用iframe或div框架集嵌入子页面设框架【设置】

    HTML5中嵌入子页面的现代方案有四种:一、用iframe标签直接嵌入,支持安全与可访问性属性;二、用CSS Grid/Flexbox布局配合JavaScript动态加载HTML片段;三、用Shadow DOM封装自定义元素实现样式脚本隔离;四、用object标签嵌入HTML并提供fallback内…

    2025年12月23日
    200
  • 带文字描边的HTML5按钮样式写法【方法】

    可通过text-shadow、-webkit-text-stroke、SVG文本或CSS自定义属性实现HTML5按钮文字描边:text-shadow兼容性好但需多向阴影;-webkit-text-stroke简洁可控但仅限WebKit浏览器;SVG提供高精度描边;CSS变量支持动态主题切换。 如果您…

    2025年12月23日
    000
  • html5怎么找颜色_html5用取色器或CSS命名如red快速找对应颜色【查找】

    可通过浏览器开发者工具取色、CSS命名颜色对照表、在线十六进制颜色查找工具及CSS自定义属性验证四种方法快速定位颜色值对应的实际色彩效果。 如果您在HTML5开发中需要快速定位某个颜色值对应的实际色彩效果,可以通过取色器工具或CSS预定义颜色名称来识别。以下是查找颜色的具体操作方法: 一、使用浏览器…

    2025年12月23日
    000
  • html5如何清除缓存_HTML5缓存清除步骤与清理浏览器缓存方法【教程】

    HTML5网页应用异常通常由浏览器缓存旧资源导致,需依次清除常规缓存、强制刷新、清理AppCache、注销Service Worker并清空其缓存、或用无痕模式验证。 如果您在使用HTML5网页应用时遇到内容未更新、页面显示异常或资源加载错误等问题,可能是由于浏览器缓存了旧版本的HTML、CSS、J…

    2025年12月23日
    000

发表回复

登录后才能评论
关注微信