当前位置: 首页 > news >正文

C++线程池介绍和C++代码实现

1、介绍

1.1 线程池应用场景

  在进行创建线程任务时,如果需要频繁的创建线程、销毁线程,这样会极大地降低效率,因为创建线程也是需要时间的,一个完整的线程处理运行时间包括:线程的创建时间、线程运作时间、线程的销毁时间。

      对于频繁创建线程的业务场景,我们可以预先创建多个线程,在创建多线程任务时,我们可以直接将线程函数添加到预先创建的线程中,这样就可以避免多次创建线程的时间,提高代码运行效率。

1.2 线程池设计的思路

        设计多线程主要实现的功能有:(1)预设创建线程的数量。(2)存储运行一定数量线程任务容器workerThread变量(可以是vector或其他容器,元素类型是std::thread),该容器一直检测任务队列变量中是否有待执行的任务,有:则取出执行,没有:则等待。(3)用于存储运行任务的变量Task(数据类型是queue,数据类型是std::function<T>),该变量主要用于存储待运行线程函数func。

线程池理论可以参考链接如下:

深入解析C++编程中线程池的使用_c++线程池用法_歌行梅村的博客-CSDN博客

1.3 线程池设计注意事项

所需理论知识

创建具有适配所有线程函数的任务队列Task,创建这样的任务队列需要有一定的C++基础,可能需要如下知识点:

可变参数模板:template <class... Args>_我在这里啊@的博客-CSDN博客

 C++多线程之旅-future等待事件_或许 没有的博客-CSDN博客

C++中函数对象模板function<T>、通用函数适配器std::bind和lambda_c++function头文件_夜雨听萧瑟的博客-CSDN博客

 C++多线程中共享变量同步问题_夜雨听萧瑟的博客-CSDN博客

1.4 预设的线程数量是多少?

        预设的线程数量不是说越多越好,而是创建适当数量的线程数,让CPU的利用率达到最大。如果预设的线程数量过大,PC的核数有限,这样同时只会有一小部分任务在同步运行,这样操作系统就需要不断的切换上下文,频繁的切换上下文也需要时间,这样反而会降低运行效率。

 经验值

主要有下面几种:

(1) 设置线程数量的一般经验值为:2N(N是CPU核数)

(2)  2N+1(N是CPU核数)

(3)  N+1(N是CPU核数)

 具体设置数量可以在设置后,对其不同数量线程数运行效率进行简单的测试。

 具体分析

可参考链接:

线程池创建线程数量讨论_夜雨听萧瑟的博客-CSDN博客

2、简单demo

(1)该demo的中心思想是创建10个运行的线程函数的容器workerThread,该线程不断检测任务队列中是否有待处理任务,有待处理任务则取出,执行任务。(2)创建一个管理任务队列数量容器Task,用户可以向其添加任务。

threadpoolm.h文件如下:

#pragma once
#include <mutex>
#include <condition_variable>
#include <thread>
#include <vector>
#include <queue>
#include <functional>class threadPoolM
{public:using funcType = std::function<void()>;threadPoolM();~threadPoolM();void setThreadNum(int num);void AddTask(const funcType& pf);
private:void StartWork();void RunTask();int m_num;std::vector<std::thread>workerThread;std::mutex mut;std::condition_variable cond;std::queue<funcType> Task;
};

threadpoolm.cpp文件如下:

#include "threadpoolm.h"
#include <iostream>
threadPoolM::threadPoolM()
{}threadPoolM::~threadPoolM()
{for(int i = 0; i < workerThread.size(); i++){workerThread.at(i).join();}
}void threadPoolM::setThreadNum(int num)
{m_num = num;StartWork();
}void threadPoolM::AddTask(const threadPoolM::funcType& pf)
{std::unique_lock<std::mutex>lk(mut);cond.wait(lk,[this]{return Task.size() < 10;});Task.push(pf);std::cout <<"id " << std::this_thread::get_id() << ", add a task, size is  " << Task.size() << std::endl;cond.notify_one();}void threadPoolM::StartWork()
{for(int i = 0; i < m_num; i++){workerThread.push_back(std::thread(&threadPoolM::RunTask,this));}
}void threadPoolM::RunTask()
{while (true) {std::unique_lock<std::mutex>lk(mut);cond.wait(lk,[this]{return Task.size()>0;});auto Ta = std::move(Task.front());Task.pop();Ta();std::cout <<"id " << std::this_thread::get_id() << ", run a task,size" << Task.size() << std::endl;cond.notify_one();}
}

上面线程池类的使用main.cpp如下:

#include <iostream>
#include "threadpoolm.h"
int cnt = 0;void printId(int id)
{std::cout << "id " << std::this_thread::get_id() << ", id" << id << std::endl;
}
int main()
{std::cout << "Hello World!" << std::endl;threadPoolM pool;pool.setThreadNum(10);while(true){if(cnt++ > 2000){cnt = 0;}std::cout << "cnt " << cnt << std::endl;auto f1 = std::bind(printId,cnt);pool.AddTask(f1); std::this_thread::sleep_for(std::chrono::milliseconds(100));//主线程中的while循环,只是对实际添加任务消息进行模拟,故没有退出while条件。}return 0;  //由于线程池中的RunTask函数存在while循环,没有退出条件,所以该行代码不会执行。可以按照实际条件对其while条件修改。
}

 运行结果如下:

 上面的代码参考于

c++11最简单的线程池实现_c++实现线程池_osDetach的博客-CSDN博客

 线程池的实现代码也可参考下面链接:

基于c++11的100行实现简单线程池_c++11 100行实现线程池 csdn_6plus的博客-CSDN博客

附加知识 

怎样理解线程的睡眠,挂起,和阻塞? - 知乎 (zhihu.com)

“阻塞(pend)”与“挂起(suspend)”的区别?_pend group run cpu调度_zhch152的博客-CSDN博客

相关文章:

C++线程池介绍和C++代码实现

1、介绍 1.1 线程池应用场景 在进行创建线程任务时&#xff0c;如果需要频繁的创建线程、销毁线程&#xff0c;这样会极大地降低效率&#xff0c;因为创建线程也是需要时间的&#xff0c;一个完整的线程处理运行时间包括&#xff1a;线程的创建时间、线程运作时间、线程的销毁…...

【day 06】vue的组件

组件 组件就是把一个网页分割成独立的小的模块&#xff0c;然后通过把模块进行组合&#xff0c;构建成一个大型的应用 单文件组件 只有一个组件 html css js 都在这个文件内 非单文件组件 可有多个组件 全局注册 !! 得先注册子组件 再生成 vm实例对象 创建子组件 const …...

第3章 Class and Object

构造函数 Guaranteed initialization with the constructor使用构造函数保证初始化 • If a class has a constructor, the compiler automatically calls that constructor at the point an object is created, before client programmers can get their hands on the o…...

卫星定位北斗芯片AT6558一款高性能BDS/GNSS多模卫星导航接收机SOC单芯片

1 芯片简介 AT6558R是一款高性能BDS/GNSS多模卫星导航接收机SOC单芯片,片上集成射频前端&#xff0c; 数字基带处理器&#xff0c;32位的RISCCPU&#xff0c;电源管理功能。 芯片支持多种卫星导航系统&#xff0c;包括中国的北斗卫星导航系统BDS&#xff0c;美国的GPS,俄罗斯 的…...

提升您的 MQTT 云服务:深入探索 BYOC

引言 您是否希望将物联网基础设施提升到更高的水平&#xff1f;为了应对业务的不断扩展&#xff0c;您需要一个强大且安全的消息平台来支持它。 MQTT 协议凭借其轻量级、发布/订阅模型和可靠性&#xff0c;已经成为构建物联网平台的首选方案。但是&#xff0c;随着业务的增长…...

Zookeeper面试题总结

1、说说 Zookeeper 是什么&#xff1f; 有些软件你想做成集群或者分布式&#xff0c;你可以用 ZooKeeper 帮你来辅助实现。特点&#xff1a;ZooKeeper 的特点&#xff1a;维护、协调、管理、监控 最终一致性&#xff1a;客户端看到的数据最终是一致的。可靠性&#xff1a;服务…...

如何使用HTML、CSS和JavaScript来制作这两种类型的时钟

随着计算机技术的不断发展和普及&#xff0c;人们对于时间的精准度要求也越来越高。时钟作为我们日常生活必不可少的工具之一&#xff0c;也得到了越来越多的关注和研究。而在Web开发中&#xff0c;我们同样可以使用HTML、CSS和JavaScript的组合&#xff0c;制作出各式各样的时…...

Java中操作Xml使用备忘

List item 文章目录 Java中操作Xml使用备忘1. Hutool中XmlUtil的使用简介2. Hutool中XmlUtil快速读取Xml字符串某个节点值 [简单取值时&#xff0c;推荐使用]2-1 Hutool工具包Maven依赖和测试Xml字符串如下2-2 读取Xml中的节点<message>的值 3. Hutool中XmlUtil详细操作示…...

【Java|基础篇】内部类

文章目录 1.什么是内部类?2.实例内部类3.静态内部类4.局部内部类5.匿名内部类6.结语 1.什么是内部类? 内部类就是在一个类中再定义一个类,内部类也是封装的体现.它可以被声明为 public、protected、private 或默认访问控制符。内部类可以访问外部类的所有成员变量和方法&…...

七牛云图床设置

文章目录 七牛云图床设置下面是用picgo配置图床SSL证书申请https网站显示http图片解决方案 原文链接图床设置&#xff0c;免费SSL证书申请&#xff0c;https网站显示http链接的图片 七牛云图床设置 登录七牛云官网并进行个人注册&#xff0c;然后找到对象存储 点击空间管理&a…...

Android 12.0下拉状态栏录屏去掉弹窗直接录屏

1.概述 在12.0的系统rom开发中,在systemui的下拉状态栏中有个录屏的快捷按钮,可以通过点击录屏实现录屏功能,但是在录屏的时候发现需要先弹出 dialog,然后点击开始实现录屏,这有的麻烦,所以需要去掉弹窗直接开始录屏,就需要弹窗的相关代码来实现功能 2.下拉状态栏录屏…...

MySql基础学习(1)

MySql基础学习 一、数据库1.1 什么是数据库1.2 MySql的启动与停止1.3 MySql数据模型 二、SQL2.1 SQL通用语法2.2 SQL分类2.2.1 数据类型2.2.2 DDL使用方法2.2.3 、表操作-修改&删除DDL总结 2.3 DML2.3.1 DML添加数据2.3.2 DML---修改数据2.3.3 DML---删除数据DML总结 2.4 D…...

18- 弹幕系统设计

1、弹幕系统设计 场景分析&#xff1a;客户端针对某一视频创建了弹幕&#xff0c;发送后端进行处理&#xff0c;后端需要对所有正在观看该视频的用户推送该弹幕。 1.1、实现方式 使用短连接进行通信或使用长连接进行通信。 1.1.1、短连接实现方案 所有观看视频的客户端不断…...

字节软测划水四年,内容过于真实......

先简单交代一下吧&#xff0c;潇潇是某不知名211的本硕&#xff0c;18年毕业加入一个小厂&#xff0c;之后跳槽到了字节跳动&#xff0c;一直从事测试开发相关的工作。之前没有实习经历&#xff0c;算是四年半的工作经验吧。 这四年半之间他完成了一次晋升&#xff0c;换了一家…...

Mybatis介绍

1. Mybatis中#和$的区别&#xff1f; #相当于对数据 加上 双引号&#xff0c;$相当于直接显示数据 1. #将传入的数据都当成一个字符串&#xff0c;会对自动传入的数据加一个双引号。如&#xff1a;order by #user_id#&#xff0c;如果传入的值是111,那么解析成sql时的值为orde…...

Docker理论基础

初识Docker 1.什么是Docker 微服务虽然具备各种各样的优势&#xff0c;但服务的拆分通用给部署带来了很大的麻烦。 分布式系统中&#xff0c;依赖的组件非常多&#xff0c;不同组件之间部署时往往会产生一些冲突。 在数百上千台服务中重复部署&#xff0c;环境不一定一致&…...

MySQL数据库之存储引擎

一、存储引擎的概念 1.1 什么是存储引擎 MySQL中的数据用各种不下同的技术存储在文件中&#xff0c;每一种技术都使用不同的存储机制、索引技巧、锁定水平并最终提供不同的功能和能力&#xff0c;这些不同的技术以及配套的功能在MySQL中称为存储引擎。存储引擎是MySQL将数据存…...

中介效应分析全流程汇总

一、中介效应说明 中介效应主要研究自变量对因变量影响的过程中&#xff0c;自变量是否通过中介变量再对因变量产生影响&#xff0c;那什么情况表明中介效应存在呢&#xff1f;如果自变量对因变量影响过程中&#xff0c;中介变量在模型中有着桥梁般的作用&#xff0c;那说明中…...

论文阅读:Multimodal Graph Transformer for Multimodal Question Answering

文章目录 论文链接摘要1 contribution3 Multimodal Graph Transformer3.1 Background on Transformers3.2 Framework overview 框架概述3.3 Multimodal graph construction多模态图的构建Text graphSemantic graphDense region graph Graph-involved quasi-attention 总结 论文…...

关于compile() 函数简单实用示例

compile() 函数是什么 compile() 函数将一个字符串编译为字节代码。 compile将代码编译为代码对象&#xff0c;应用在代码中可以提高效率。 语法 compile(source, filename, mode, flags0, dont_inheritFalse, optimize-1) 参数 source&#xff1a;表示要编译的源代码字符串、…...

戴森球计划工厂蓝图架构深度解析:构建高效星际生产线的核心策略

戴森球计划工厂蓝图架构深度解析&#xff1a;构建高效星际生产线的核心策略 【免费下载链接】FactoryBluePrints 游戏戴森球计划的**工厂**蓝图仓库 项目地址: https://gitcode.com/GitHub_Trending/fa/FactoryBluePrints FactoryBluePrints 项目作为戴森球计划游戏中最…...

Qwen-Image-2512+LoRA:构建Godot原生像素素材生成管线

1. 这不是“AI画图”&#xff0c;而是一次像素艺术工作流的底层重写你有没有试过在Godot 4.x里导入一张用Qwen-VL或Stable Diffusion生成的“像素风”图&#xff1f;放大一看——边缘糊成一团&#xff0c;颜色溢出格子&#xff0c;连88的精灵都对不齐网格。我去年帮一个独立游戏…...

如何快速构建数学可视化:Manim交互式开发完整教程

如何快速构建数学可视化&#xff1a;Manim交互式开发完整教程 【免费下载链接】manim Animation engine for explanatory math videos 项目地址: https://gitcode.com/GitHub_Trending/ma/manim 想要告别数学动画制作中反复修改代码、重新渲染的烦恼吗&#xff1f;&…...

MoE稀疏激活原理与实战:解密大模型高效计算的核心机制

1. 这不是“参数越多越强”的简单故事&#xff1a;拆解大模型里那个被悄悄藏起来的“开关”你肯定见过这类标题&#xff1a;“GPT-4 参数量突破1.8万亿&#xff01;”、“DeepSeek-R1 达到6710亿参数&#xff01;”——光看数字&#xff0c;像在比谁家粮仓堆得更高。但真正懂行…...

3步快速上手:Windows安卓应用安装器的终极指南

3步快速上手&#xff1a;Windows安卓应用安装器的终极指南 【免费下载链接】APK-Installer An Android Application Installer for Windows 项目地址: https://gitcode.com/GitHub_Trending/ap/APK-Installer 你是否曾想在Windows电脑上直接运行安卓应用&#xff1f;告别…...

微软Windows拆分:云AI战略转型下的业务重构与行业影响

1. 从“巨无霸”到“手术台”&#xff1a;微软拆分的深层逻辑与行业变局最近几年&#xff0c;关于微软可能进行业务拆分的讨论&#xff0c;就像科技行业的“月经帖”&#xff0c;每隔一段时间就会冒出来。但这一次&#xff0c;市场的风声似乎比以往任何时候都要紧。从“拆分Win…...

海思HS-Ultra 7L2D边缘AI工控机:99 TOPS算力如何赋能工业视觉与医疗?

1. 项目概述&#xff1a;当AI算力下沉到边缘&#xff0c;我们需要什么样的“工业大脑”&#xff1f;在工业自动化、机器视觉、智慧医疗这些领域摸爬滚打十几年&#xff0c;我亲眼见证了计算架构从集中式云端向边缘侧迁移的浪潮。早期&#xff0c;大家习惯于把海量的传感器数据、…...

Apache Camel实战指南:从企业集成到微服务架构的完美演进

Apache Camel实战指南&#xff1a;从企业集成到微服务架构的完美演进 【免费下载链接】camelinaction2 :camel: This project hosts the source code for the examples of the Camel in Action 2nd ed book :closed_book: written by Claus Ibsen and Jonathan Anstey. 项目地…...

如何快速部署大麦自动抢票工具:面向开发者的完整技术指南

如何快速部署大麦自动抢票工具&#xff1a;面向开发者的完整技术指南 【免费下载链接】ticket-purchase 大麦自动抢票&#xff0c;支持人员、城市、日期场次、价格选择 项目地址: https://gitcode.com/GitHub_Trending/ti/ticket-purchase 在热门演出票务市场中&#xf…...

告别手速焦虑:大麦抢票自动化系统全攻略

告别手速焦虑&#xff1a;大麦抢票自动化系统全攻略 【免费下载链接】ticket-purchase 大麦自动抢票&#xff0c;支持人员、城市、日期场次、价格选择 项目地址: https://gitcode.com/GitHub_Trending/ti/ticket-purchase 还在为抢不到演唱会门票而烦恼吗&#xff1f;每…...