简易版的进程池

2024-06-22 05:28
文章标签 进程 简易版

本文主要是介绍简易版的进程池,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.使用匿名管道实现 进程池

#pragma oncetypedef void(*task_t)();void task1()
{cout << "游戏 刷新日志" << endl;
}void task2()
{cout << "游戏 刷新野区" << endl;
}
void task3()
{cout << "游戏 检测软件是否更新,如果需要,就提示用户" << endl;
}
void task4()
{cout << "游戏 用户释放技能,更新用户的HP和MP"<<endl;
}
void Loadtask(vector<task_t> *tasks)
{tasks->push_back(task1);tasks->push_back(task2);tasks->push_back(task3);tasks->push_back(task4);
}//以上是头文件//以下是程序;
//类似于内存池,每次使用都不再向系统申请进程,而是提前储备进程;
#include<cstdlib>
#include<cassert>
#include<time.h>
#include<sys/wait.h>
#include<sys/stat.h>
#include<iostream>
#include<vector>
#include<string>
#include<unistd.h>using namespace std;
#include"main.h"
const int processnum = 5;//子进程数量=5
vector<task_t> tasks;  //任务列表,全局变量//先描述
class channel
{
public:channel(int cmdfd ,pid_t slaverid,const string &processname):_cmdfd(cmdfd),_slaverid(slaverid),_processname(processname){}
public:int _cmdfd;	//发送任务的文件描述符pid_t _slaverid; //子进程的pidstring _processname;//子进程的名字——方便我们打印日志;
};void slaver()
{//read(0);//直接从标准输入读while (true){int cmdcode = 0;int n = read(0,&cmdcode,sizeof(int));//如果父进程不给子进程发任务?就会阻塞等待;if (n == sizeof(int)){//执行cmdcode 对应的任务列表;cout << getpid() << ":cmdcode:" << cmdcode << endl;//查看父进程给子进程派发的任务码;if (cmdcode > 0 && cmdcode < tasks.size()) //满足条件运行函数;tasks[cmdcode]();}if (n == 0){break;}}
}
//函数传参,需要输入型,输出型,输入输出型参数;
//输入:const &: const引用
//输出:用指针
//输入输出:&引用
void InitProcessPool(vector<channel>* channels)
{//1.初始化for (int i = 0; i < processnum; i++){int pipefd[2];//临时空间;int n = pipe(pipefd);assert(!n);//判断是否成功;(void)n;pid_t id = fork();if (id == 0)//子进程读{close(pipefd[1]);dup2(pipefd[0], 0);//重定向,从标准输入中读close(pipefd[0]);slaver();//执行相对应的任务;cout << "process: " << getpid() << "quit" << endl;//打印子进程pid;退出进程;exit(0);}//父进程写入close(pipefd[0]);//添加channel字段;string name = "process-" + to_string(i);channels->push_back(channel(pipefd[1], id, name));}
}
void Debug(const vector<channel> &channels)
{//testfor (const auto& c : channels){cout << c._cmdfd << " " << c._slaverid << " " << c._processname << endl;}
}
void Menu()											//菜单
{cout << "1.刷新日志" << endl;cout << "2.刷新野怪" << endl;cout << "3.检测更新" << endl;cout << "4.更新状态" << endl;cout << "0.退出游戏" << endl;
}
//控制子进程
void ctrlSlaver(const vector<channel>& channels)
{int which = 0;//int cnt = 5;while(true){int select = 0;Menu();                              //手动控制cout << "please Enter@" << endl;cin >> select;if (select <= 0 || select >= 5) break;//select >0 &&select<5//1.选择任务//int cmdcode = rand() % tasks.size();//创造一个随机数,当作任务;int cmdcode = select-1;//2.选择进程(负载均衡:不可以一直给某个进程派发任务,其他进程都闲着,要均衡发展这个叫做负载均衡)1.使用随机数  2.轮转l来实现负载均衡//int processpos = rand() % channels.size();						//随机数//cout << "father say:" << "cmdcode:" << cmdcode << "already send to" << channels[processpos]._slaverid << endl;cout << "father say:" << "cmdcode:" << cmdcode << "already send to" << channels[which]._slaverid << endl;//轮转//3.发送任务//write(channels[processpos]._cmdfd, &cmdcode, sizeof(cmdcode));      //随机数版本write(channels[which]._cmdfd, &cmdcode, sizeof(cmdcode));			  //轮转which++;which %= channels.size();//cnt--;//sleep(1);}
}
void QuitProcess(const vector<channel> &channels)						//退出进程
{for (const auto& c : channels){close(c._cmdfd);}//sleep(5);for (const auto& c : channels) waitpid(c._slaverid, nullptr, 0);   //等待退出//sleep(5);
}int main()
{Loadtask(&tasks);							//加载任务参数;srand(time(nullptr) ^ getpid() ^ 1023);		//种一个随机数种子//再组织vector<channel> channels;//初始化-------bug?————画图就可以看出来,推荐画一下图!!!答:文件描述符随着创建子进程递增;InitProcessPool(&channels);					//初始化Debug(channels);							//测试//2.开始控制子进程ctrlSlaver(channels);						//控制子进程//3.清理收尾QuitProcess(channels);return 0;
}

这篇关于简易版的进程池的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1083410

相关文章

Windows的CMD窗口如何查看并杀死nginx进程

《Windows的CMD窗口如何查看并杀死nginx进程》:本文主要介绍Windows的CMD窗口如何查看并杀死nginx进程问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录Windows的CMD窗口查看并杀死nginx进程开启nginx查看nginx进程停止nginx服务

Java进程CPU使用率过高排查步骤详细讲解

《Java进程CPU使用率过高排查步骤详细讲解》:本文主要介绍Java进程CPU使用率过高排查的相关资料,针对Java进程CPU使用率高的问题,我们可以遵循以下步骤进行排查和优化,文中通过代码介绍... 目录前言一、初步定位问题1.1 确认进程状态1.2 确定Java进程ID1.3 快速生成线程堆栈二、分析

Linux实现简易版Shell的代码详解

《Linux实现简易版Shell的代码详解》本篇文章,我们将一起踏上一段有趣的旅程,仿照CentOS–Bash的工作流程,实现一个功能虽然简单,但足以让你深刻理解Shell工作原理的迷你Sh... 目录一、程序流程分析二、代码实现1. 打印命令行提示符2. 获取用户输入的命令行3. 命令行解析4. 执行命令

Python多进程、多线程、协程典型示例解析(最新推荐)

《Python多进程、多线程、协程典型示例解析(最新推荐)》:本文主要介绍Python多进程、多线程、协程典型示例解析(最新推荐),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定... 目录一、multiprocessing(多进程)1. 模块简介2. 案例详解:并行计算平方和3. 实现逻

C#通过进程调用外部应用的实现示例

《C#通过进程调用外部应用的实现示例》本文主要介绍了C#通过进程调用外部应用的实现示例,以WINFORM应用程序为例,在C#应用程序中调用PYTHON程序,具有一定的参考价值,感兴趣的可以了解一下... 目录窗口程序类进程信息类 系统设置类 以WINFORM应用程序为例,在C#应用程序中调用python程序

Python如何精准判断某个进程是否在运行

《Python如何精准判断某个进程是否在运行》这篇文章主要为大家详细介绍了Python如何精准判断某个进程是否在运行,本文为大家整理了3种方法并进行了对比,有需要的小伙伴可以跟随小编一起学习一下... 目录一、为什么需要判断进程是否存在二、方法1:用psutil库(推荐)三、方法2:用os.system调用

Java程序进程起来了但是不打印日志的原因分析

《Java程序进程起来了但是不打印日志的原因分析》:本文主要介绍Java程序进程起来了但是不打印日志的原因分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java程序进程起来了但是不打印日志的原因1、日志配置问题2、日志文件权限问题3、日志文件路径问题4、程序

Linux中的进程间通信之匿名管道解读

《Linux中的进程间通信之匿名管道解读》:本文主要介绍Linux中的进程间通信之匿名管道解读,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、基本概念二、管道1、温故知新2、实现方式3、匿名管道(一)管道中的四种情况(二)管道的特性总结一、基本概念我们知道多

Linux进程终止的N种方式详解

《Linux进程终止的N种方式详解》进程终止是操作系统中,进程的一个重要阶段,他标志着进程生命周期的结束,下面小编为大家整理了一些常见的Linux进程终止方式,大家可以根据需求选择... 目录前言一、进程终止的概念二、进程终止的场景三、进程终止的实现3.1 程序退出码3.2 运行完毕结果正常3.3 运行完毕

Windows命令之tasklist命令用法详解(Windows查看进程)

《Windows命令之tasklist命令用法详解(Windows查看进程)》tasklist命令显示本地计算机或远程计算机上当前正在运行的进程列表,命令结合筛选器一起使用,可以按照我们的需求进行过滤... 目录命令帮助1、基本使用2、执行原理2.1、tasklist命令无法使用3、筛选器3.1、根据PID