【Linux】进程通信实战 —— 进程池项目

在这里插入图片描述
送给大家一句话:
没有一颗星,会因为追求梦想而受伤,当你真心渴望某样东西时,整个宇宙都会来帮忙。 – 保罗・戈埃罗 《牧羊少年奇幻之旅》

🏕️🏕️🏕️🏕️🏕️🏕️
🗻🗻🗻🗻🗻🗻


进程通信实战 —— 进程池项目

  • 1 ♻️知识回顾
  • 2 ♻️项目介绍
  • 3 ♻️项目实现
    • 3.1 ✨创建信道和子进程
    • 3.2 ✨建立任务
    • 3.3 ✨控制子进程
    • 3.4 ✨回收信道和子进程
  • 4 ♻️总结
  • Thanks♪(・ω・)ノ谢谢阅读!!!
  • 下一篇文章见!!!

1 ♻️知识回顾

在之前的讲解中,我们深入探讨了以下几个方面:

  1. 父子进程的创建与管理:我们详细讲解了父子进程是如何建立的,以及子进程如何继承父进程的代码和数据。子进程通常用于完成特定的任务。
  2. 文件操作:我们学习了如何使用 read 和 write 操作文件,并了解了文件描述符(fd)的概念,从而能够在文件中进行信息的读取和写入。
  3. 进程间通信:我们介绍了匿名管道,这是一种父子进程间进行通信的方式。通过共享资源,父子进程可以实现数据的传递和同步。

在接下来的内容中,让我们把所学知识来进行运用,我们将探讨进程池的概念和实现细节。

2 ♻️项目介绍

进程池是一种用于管理和复用进程的技术,它可以有效地管理系统资源并提高程序的性能和效率。通过维护一组预先创建的进程与管道,进程池可以避免频繁地创建和销毁进程,从而减少了系统开销和资源浪费。
在这里插入图片描述
主要使用的是池化技术的思想:

池化技术是一种广泛应用于系统开发中的优化策略,旨在通过复用资源来提高性能和效率。池化技术的核心思想是预先分配一组资源,并在需要时进行复用,而不是每次都重新创建和销毁资源。

池化技术(Pooling)涉及创建和管理一组预先分配的资源,这些资源可以是进程、线程、数据库连接或对象实例。在池化系统中,当请求到达时,它会从池中获取一个空闲资源,使用完毕后将其归还池中。这种方法避免了频繁的创建和销毁操作,从而显著减少了系统开销。

进程池就是通过预先创建若干个进程与管道,在需要进行任务时,选择一个进程,通过管道发送信息,让其完成工作。

进程池在实际项目中有广泛的应用,尤其是在处理大量并发任务时,例如:网络服务器中的请求处理、数据处理以及计算密集型任务。通过合理配置进程池的大小和参数,可以有效控制系统负载,提高整体响应速度。

3 ♻️项目实现

3.1 ✨创建信道和子进程

首先我们需要建立一个信道类,来储存管道及其对应的子进程信息。

//信道类
class Channel
{
public:Channel(pid_t id , int wfd , std::string name):_id(id) , _wfd(wfd) , _name(name)    {}~Channel(){}void Close(){close(_wfd);}//关闭管道时需要等待对应子进程结束void WaitSub(){pid_t rid = waitpid(_id, nullptr, 0);if (rid > 0){std::cout << "wait " << rid << " success" << std::endl;}}pid_t GetId(){ return _id;}int GetWfd(){ return _wfd;}std::string GetName(){return _name ;}
private:pid_t _id ;//对应 子进程 idint _wfd  ;//写入端std::string _name ; //管道名称
};

然后我们就建立若干个信道与子进程,创建子进程与信道的时候,把信息插入到信道容器中,完成储存。子进程需要阻塞在读取文件,等待父进程写入信息:

void CreateChannel(int num , std::vector<Channel>* channel)
{//初始化任务InitTask();for(int i = 0 ; i < num ; i++){//创建管道int pipefd[2] = {0};int n = pipe(pipefd);if(n != 0){std::cout << "create pipe failed!" << std::endl;}//创建子进程pid_t id = fork();if(id == 0){//子进程 --- 只读不写close(pipefd[1]);work(pipefd[0]);close(pipefd[0]);exit(0);}//父进程close(pipefd[0]);std::string name = "Channel - " + std::to_string(i);//储存信道信息channel->push_back( Channel(id , pipefd[1] , name) );}
}

这里提一下传参的规范:

  • const &:表示输出型参数,即该参数是输入型,不会被修改。常用于传递不需要修改的对象或数据。
  • &:表示输入输出型参数,即该参数既是输入参数,又是输出参数,函数可能修改其内容。
  • * :表示输出型参数,通常用于传递指针,函数通过指针参数返回结果给调用者。

进行一下测试,看看是否可以这正常建立信道与子进程;

int main(int argc , char* argv[])
{//1. 通过main函数的参数 int argc char* argv[] (./ProcessPool 5) //判断要创建多少个进程if(argc != 2){std::cout << "请输入需要创建的信道数量 :"  << std::endl;}std::vector<Channel> channel;int num = std::stoi(argv[1]);//2. 创建信道和子进程CreateChannel(num , &channel);//测试:for(auto t :  channel){std::cout<< "==============="<<std::endl;std::cout<< "信道对应 name :" << t.GetName() <<std::endl;std::cout<< "信道对应子进程 pid :" << t.GetId() <<std::endl;std::cout<< "信道对应写端 wfd :" << t.GetWfd() <<std::endl;}return 0;
}

在这里插入图片描述
完美,可以正常创建!!!

3.2 ✨建立任务

完成了信道与子进程的创建,接下来我们就来设置一些任务。我们在.hpp文件里直接把声明定义写在一起,确保代码的模块化和可维护性。

void Print()
{std::cout << "this is Print()"<< std::endl;
}void Fflush()
{std::cout << "this is Fflush()"<< std::endl;
}void Scanf()
{std::cout << "this is Scanf()"<< std::endl;
}

然后通过函数指针数来储存这些函数,因为子进程会继承父进程的数据,这样通过一个数字下标即可确定调用的函数。只需要传入 4 个字节的int类型,最大程度的减少了通信的成本!!!

#pragma once#include <iostream>
#include <ctime>
#include <cstdlib>
#include <sys/types.h>
#include <unistd.h>#define TaskNum 3
//这个文件里是任务函数
typedef void(*task_t)();task_t tasks[TaskNum];
//...
//...三个函数
//...
void InitTask()
{srand(time(nullptr) ^ getpid() ^ 17777);tasks[0] = Print;tasks[1] = Fflush;tasks[2] = Scanf;
}
//执行任务!!!
void ExecuteTask(int num)
{if(num < 0 || num > 2) return;tasks[num]();
}
//随机挑选一个任务
int SelectTask()
{return rand() % TaskNum;
}

3.3 ✨控制子进程

首先通过 SelectTask() 选择一个任务,然后选择一个信道和子进程。需要注意的是,这里要依次调用每一组子进程,采用轮询(Round-Robin)方案,以尽可能实现负载均衡。 然后发送任务(向信道写入4字节的数组下标)

int SelectChannel(int n)
{//静态变量做到轮询方案static int next = 0;int channel = next;next++;next %= n;return channel;
}
void SendTaskCommond(Channel& channel , int TaskCommand  )
{//写入对应信息write(channel.GetWfd() , &TaskCommand , sizeof(TaskCommand));
}void CtrlProcessOnce(std::vector<Channel>& channel)
{//选择一个任务int TaskCommand = SelectTask();//选择一个进程与信道int ChannelNum = SelectChannel(channel.size());//发送信号//测试std::cout << "taskcommand: " << TaskCommand << " channel: "<< channel[ChannelNum].GetName() << " sub process: " << channel[ChannelNum].GetId() << std::endl; SendTaskCommond(channel[ChannelNum] ,TaskCommand);}

我们写入之后,子进程就可以读取任务并执行,注意子进程读取只读4个字节!!!如果读取的个数不正确,那么就出现了错误,需要报错!!!

//子进程运行函数
void work(int rfd)
{while(true){int Commond = 0;//等待相应int n = read(rfd , &Commond , sizeof(Commond));if(n == sizeof(int)){std::cout << "pid is : " << getpid() << " handler task" << std::endl;//执行命令std::cout << "commond :" << Commond << std::endl;ExecuteTask(Commond);}//写端关闭else if(n == 0){std::cout << "sub Process:" << getpid() << std::endl;break;}}
}
//...//创建子进程pid_t id = fork();if(id == 0){//子进程 --- 只读不写close(pipefd[1]);work(pipefd[0]);close(pipefd[0]);exit(0);}
//...

进行一下测试:
在这里插入图片描述
成功执行任务!!!

3.4 ✨回收信道和子进程

首先关闭信道写端,这样子进程会自己退出,然后父进程等待子进程退出(wait等待子进程 )不要出现僵尸进程 !!!

注意由于子进程会继承父进程的数据,所以一个信道实际上会有多个写端。为了不必要的错误,分开集中操作:先关闭所有写端,再等待所以子进程。

void CleanUpChannel(std::vector<Channel>& channel)
{for(auto t : channel){t.Close();}for(auto t : channel){t.WaitSub();}
}

测试一下:
在这里插入图片描述
5 个子进程成功退出释放!!!

4 ♻️总结

这样,我们的进程池项目就完成了。不过,实际上我们还可以进一步优化,比如优化 work 函数,将其设置为回调函数,以实现完全解耦。

尽管如此,目前的实现已经能够满足我们的项目需求。一个面向过程的进程池项目就此完成!!!

Thanks♪(・ω・)ノ谢谢阅读!!!

下一篇文章见!!!

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

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

相关文章

flink cdc mysql整理与总结

文章目录 一、业务中常见的需要数据同步的场景CDC是什么FlinkCDC是什么CDC原理为什么是FlinkCDC业务场景flink cdc对应flink的版本 二、模拟案例1.阿里云flink sql2.开源flink sql(单机模式)flink 安装安装mysql3.flink datastream 三、总结 提示&#xff1a;以下是本篇文章正文…

kali基本扫描工具(自带)

免责声明:本文仅做技术交流与学习...请勿非法破坏... 详细用法: 命令 -h/百度/翻译 fping 用法 hostlist 文件里面为ip fping -a -q -f hostlist -a 只看存活的 fping -g 202.100.1.1 202.100.1.255 -a -q > Ahost 输出到Ahost文件上 nping nping -c 1 201.100.2.155-244 …

香橙派AIpro使用SSH远程登录

香橙派AIpro可以连接HDMI显示器使用&#xff0c;也可以远程登录。这里采用MobaXterm软件远程登录开发板。 首先要使得控制电脑和香橙派开发板连接到同一个局域网&#xff0c;两者的IP地址能够ping通。在Windows 下可以使用MobaXterm 远程登录开发板&#xff0c;首先新建一个ss…

属于程序员的浪漫,一颗会跳动的心!!!

绘制一颗会跳动的心❤ 嘿嘿 可以说是程序员的专属浪漫了吧&#xff0c;就像点燃一颗LED灯一样&#xff1f;&#xff08;我瞎说的啊&#xff0c;大家别当真&#xff0c;我很菜的&#xff01;&#xff01;&#xff01;&#xff01;&#xff09; 程序就在下面啦&#xff0c;然…

Java8Stream

目录 什么是Stream? IO流&#xff1a; Java8Stream&#xff1a; 什么是流&#xff1f; stream图解 获取流 集合类&#xff0c;使用 Collection 接口下的 stream() 代码 数组类&#xff0c;使用 Arrays 中的 stream() 方法 代码 stream&#xff0c;使用 Stream 中的…

重生之 SpringBoot3 入门保姆级学习(02、打包部署)

重生之 SpringBoot3 入门保姆级学习&#xff08;02、打包部署&#xff09; 1.6 打包插件1.7 测试 jar 包1.8 application.properties 的相关配置 1.6 打包插件 官网链接 https://docs.spring.io/spring-boot/docs/current/reference/html/getting-started.html#getting-starte…

【Python】 XGBoost模型的使用案例及原理解析

原谅把你带走的雨天 在渐渐模糊的窗前 每个人最后都要说再见 原谅被你带走的永远 微笑着容易过一天 也许是我已经 老了一点 那些日子你会不会舍不得 思念就像关不紧的门 空气里有幸福的灰尘 否则为何闭上眼睛的时候 又全都想起了 谁都别说 让我一个人躲一躲 你的承诺 我竟然没怀…

Kubernetes(K8S) 集群环境搭建指南

Kubernetes&#xff08;简称K8s&#xff09;是一个开源的容器编排平台&#xff0c;旨在自动化部署、扩展和管理容器化应用。K8S环境搭建过程比较复杂&#xff0c;涉及到非常多组件安装和系统配置&#xff0c;本文将会详细介绍如何在服务器上搭建好Kubernetes集群环境。 在学习…

面试题·栈和队列的相互实现·详解

A. 用队列实现栈 用队列实现栈 实现代码如下 看着是队列&#xff0c;其实实际实现更接近数组模拟 typedef struct {int* queue1; // 第一个队列int* queue2; // 第二个队列int size; // 栈的大小int front1, rear1, front2, rear2; // 两个队列的首尾指针 } MyS…

6.Redis之String命令

1.String类型基本介绍 redis 所有的 key 都是字符串, value 的类型是存在差异的~~ 一般来说,redis 遇到乱码问题的概率更小~~ Redis 中的字符串,直接就是按照二进制数据的方式存储的. (不会做任何的编码转换【讲 mysql 的时候,知道 mysql 默认的字符集, 是拉丁文,插入中文…

Jenkins--从入门到入土

Jenkins–从入门到入土 文章目录 Jenkins--从入门到入土〇、概念提要--什么是CI/DI&#xff1f;1、CI&#xff08;Continuous Integration&#xff0c;持续集成&#xff09;2、DI&#xff08;DevOps Integration&#xff0c;DevOps 集成&#xff09;3、解决的问题 一、Jenkins安…

world machine学习笔记(4)

选择设备&#xff1a; select acpect&#xff1a; heading&#xff1a;太阳的方向 elevation&#xff1a;太阳的高度 select colour&#xff1a;选择颜色 select convexity&#xff1a;选择突起&#xff08;曲率&#xff09; select height&#xff1a;选择高度 falloff&a…

用常识滚雪球:拼多多的内生价值,九年的变与不变

2024年5月22日&#xff0c;拼多多公布了今年一季度财报&#xff0c;该季度拼多多集团营收868.1亿元&#xff0c;同比增长131%&#xff0c;利润306.0亿&#xff0c;同比增长了202%&#xff0c;数据亮眼。 市场对拼多多经历了“看不见”、“看不懂”、“跟不上”三个阶段。拼多多…

Vue.js条件渲染与列表渲染指南

title: Vue.js条件渲染与列表渲染指南 date: 2024/5/26 20:11:49 updated: 2024/5/26 20:11:49 categories: 前端开发 tags: VueJS前端开发数据绑定列表渲染状态管理路由配置性能优化 第1章&#xff1a;Vue.js基础与环境设置 1.1 Vue.js简介 Vue.js (读音&#xff1a;/vju…

SwiftUI中的Slider的基本使用

在SwiftUI中&#xff0c;可以使用Slider视图创建一个滑动条&#xff0c;允许用户从范围中选择一个值。通过系统提供的Slider&#xff0c;用起来也很方便。 Slider 先看一个最简单的初始化方法&#xff1a; State private var sliderValue: Float 100var body: some View {V…

ollama 使用,以及指定模型下载地址

ollama windows 使用 官网&#xff1a; https://ollama.com/ windows 指定 models 下载地址 默认会下载在C盘 &#xff0c;占用空间 在Windows系统中&#xff0c;可以通过设置环境变量OLLAMA_MODELS来指定模型文件的下载和存储路径。具体操作步骤如下&#xff1a; 1.打开系统…

【九十四】【算法分析与设计】练习四蛮力法练习,排列问题和组合问题,求解最大连续子序列和问题,求解幂集问题,求解0/1背包问题,求解任务分配问题

求解最大连续子序列和问题 给定一个有n&#xff08;n≥1&#xff09;个整数的序列&#xff0c;要求求出其中最大连续子序列的和。 例如&#xff1a; 序列&#xff08;-2&#xff0c;11&#xff0c;-4&#xff0c;13&#xff0c;-5&#xff0c;-2&#xff09;的最大子序列和为20…

第 33 次CCF认证

1. 词频统计 题目描述 样例输入 代码 #include <bits/stdc.h>using namespace std;int main() {int n,m;cin>>n>>m;vector<int> ans1(m,0),ans2(m,0);while (n --) {int t;cin>>t;vector<int> vis(m1,0);for (int i 1;i < t;i ) {i…

数据结构(五)

数据结构&#xff08;五&#xff09; 常见的排序算法内部排序交换插入选择归并基数 外部排序基于归并的 常见的排序算法 内部排序 交换 冒泡&#xff1a;每一次运行总会将最小的或者最大的放到前面&#xff0c;如果需要交换&#xff0c;一直在交换 快速排序*&#xff1a;经过…

2024最新前端面试八股文【基础篇293题】

⼀、HTML、HTTP、web综合问题 1 前端需要注意哪些SEO 2 <img> 的 title 和 alt 有什么区别 3 HTTP的⼏种请求⽅法⽤途 4 从浏览器地址栏输⼊url到显示⻚⾯的步骤 5 如何进⾏⽹站性能优化 6 HTTP状态码及其含义 7 语义化的理解 8 介绍⼀下你对浏览器内核的理解 9 …