【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;以下是本篇文章正文…

mac中文件夹怎么显示.git隐藏文件

1. 打开终端应用程序&#xff0c;然后进入到包含.git文件夹的目录&#xff0c;可以使用以下命令来显示隐藏文件和文件夹&#xff1a; defaults write com.apple.finder AppleShowAllFiles YES 2. 然后重启 Finder&#xff1a; killall Finder

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 …

工具方法 - 如何在网上找资料

在查询USB相关的技术资料时&#xff0c;官网的文档中心里找到个spec的记录&#xff0c;但下载链接没有。然后在Google上搜索&#xff1a; fileytpe:pdf my_keyword 只找到一个收费的文档下载网站&#xff0c;这让我不开心。 于是在Yandex上搜了下&#xff0c;找到了两个网站可以…

香橙派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;然…

hive结合Hbase实现实时数据处理和批量分析

问题背景 Hive主要设计为一个用于大数据集的批处理查询引擎&#xff0c;并不是为实时查询或实时数据更新而设计的。它主要用于执行数据摘要、查询和分析。因此&#xff0c;Hive本身不支持实时数据更新或实时查询&#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模型的使用案例及原理解析

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

自学动态规划—— 一和零

一和零 474. 一和零 - 力扣&#xff08;LeetCode&#xff09; 其实遇到这种还好说&#xff0c;我宁愿遇见这种&#xff0c;也不想遇见那些奇奇怪怪递推公式的题目。 这里其实相当背包要满足两个条件&#xff0c;所以我们可以将dp开成二维的&#xff0c;之后的操作&#xff0…

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

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

C语言---求一个整数存储在内存中的二进制中1的个数--3种方法

//编写代码实现&#xff1a;求一个整数存储在内存中的二进制中1的个数 //第一种写法 /*int count_bit_one(unsigned int n) {int count 0;while (n )//除到最后余数是0&#xff0c;那么这个循环就结束了{//这个题就是可以想成求15的二进制的过程//每次都除以2&#xff0c;余数…

跟小伙伴们说一下

因为很忙&#xff0c;有一段时间没有更新了&#xff0c;这次先把菜鸟教程停更一下&#xff0c;因为自己要查缺补漏一些细节问题&#xff0c;而且为了方便大家0基础也想学C语言&#xff0c;这里打算给大家开一个免费专栏&#xff0c;这里大家就可以好好学习啦&#xff0c;哪怕0基…

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

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

图像处理ASIC设计方法 笔记25 红外成像技术:未来视觉的革命

在当今科技飞速发展的时代,红外成像技术以其独特的优势,在医疗、工业检测等多个领域扮演着越来越重要的角色。本章节(P146 第7章红外焦平面非均匀性校正SoC)将深入探讨红外成像系统中的关键技术——非均匀性校正SoC,以及它如何推动红外成像技术迈向新的高度。 红外成像系统…

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安…

iOS 开发系列:基于VNRecognizeTextRequest识别图片文字

1.添加Vision Kit依赖 在项目设置中点击"General"选项卡&#xff0c;然后在"Frameworks, Libraries, and Embedded Content"&#xff08;框架、库和嵌入内容&#xff09;部分&#xff0c;点击""按钮。搜索并选择"Vision.framework"。…

[AIGC] flink sql 消费kafka消息,然后写到mysql中的demo

这是一个使用 Flink SQL 从 Kafka 中消费数据并写入 MySQL 的示例。在这个示例中&#xff0c;我们将假设有一个 Kafka 主题 “input_topic”&#xff0c;它产生格式为 (user_id: int, item_id: int, behavior: string, timestamp: long) 的数据&#xff0c;我们需要把这些数据写…