您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > C/C++/C#

C++实现线程池

时间:2021-05-14 13:42:11  来源:微信公众号  作者:小懵白

在上一篇文章C++使用socket实现与微信小程序通信(下)中,小懵白就给大家简要地讲解了线程池的原理。

C++实现线程池

 

今天呢,小懵白就给大家继续讲解C++如何实现封装线程池类。

第一步

首先,我们需要定义生产者、消费者的存储容器类型。

在这里呢,消费者的容器是通过使用SLT中的vector容器来实现的:std::vector<std::thread> pool;

而任务生产者容器则是由SLT的queue容器来实现:std::queue<Task>task ;

C++实现线程池

 

定义完生产者、消费者的容器类型后,接着就是要构建消费者容器:往pool里添加一定数量N的阻塞线程。

C++实现线程池

 

简要代码如下:

bool Init_ThreadPool(){
  for(int i=0;i<max_thread;i++)  //max_thread为初始化线程最大化量     
        pool.push_back(std::thread(&ThreadPool::runtask,this));
    return true;
}

在初始化中,每一个线程都绑定了runtask函数,也就是线程要执行的函数,该函数是用来实现获取task的。

 

简要代码如下:

void runtask(){
    
  while(!stop){
      
      std::unique_lock<std::mutex>lk(_mutex);
    
      //当任务队列为空或者stop为假时,阻塞当前线程,直到条件变量唤醒 
      condition.wait(lk,[this]{return (!task.empty()||stop);});  
        
      /**********************************************
       std::move是将对象的状态或者所有权从一个对象转移到另一个对象,
        只是转移,没有内存的搬迁或者内存拷贝。
      ***********************************************/ 
       Task ta=move(task.front());// 取一个 task
       task.pop();  //从队列移除正在执行的任务
       lk.unlock();   
       ta();  //执行task任务 
       condition.notify_one(); //通知等待一个线程
  }
  //此处补充销毁线程代码
}

在该代码中,bool型的stop变量是用来判定线程是否终止的,每当一次while循环后进行判断stop,若为假则继续阻塞线程然后获取任务,否则的话结束线程。

消费者实现后,接下来就是实现生产者了。在此之前,还需要做一件事情:同一规划任务变量。

using Task = std::function<void()>;

这是由于每一个task函数的返回类型都不可能是统一的,有的是void类型的,有的是bool类型的,更还有的是指针类型的。

为了能够实现代码的高效性,这里采用了std::function<void()>思想,定义了Task类型。

接着就是实现生产者了:

bool Add_task(const Task&t){
  
  if(task.size()==max_queue)Warn_LOG("添加任务,任务队列已满,准备阻塞");
  if(thread_run==max_thread)Warn_LOG("添加任务,但线程已经分配完");
  
  std::unique_lock<std::mutex>lk(_mutex);
            
    while(task.size()==max_queue||stop){ //要是任务数量到了最大,就等待处理完再添加
            condition.wait(lk);  
        }
    if(stop)return false;
      
    task.push(t);//给队列中添加任务
    task_numble=task.size();
        
    std::cout<<"添加成功。"<<std::endl<<std::endl;
    condition.notify_one(); //通知等待一个线程 
    return true;
}

为此,到这里的时候,生产者和消费者的线程池初步构建完成了

 

第二步:解说

在第一步中,我们通过使用STL中的vector容器存储了已初始化的线程后,那么接下来是如何实现生产者-消费者模式呢。

在生产者pool容器中,每一个线程都是绑定了runtask函数,也就是说每一个线程都是在执行runtask函数。

C++实现线程池

 

而在runtask函数代码块里,首先是判断!stop条件是否满足,若不满足,则结束线程。

否则的话,添加互斥锁,若任务队列为空或者stop为假时,使用条件锁std::condition_variable阻塞当前线程。

直到条件变量唤醒后,才从任务队列中获取一个task,解锁互斥锁并执行task,直至完成并通知等待一个线程,然后重新循环判断。

第三步:封装

实现完上面的功能后,接下来就是对功能进行封装了。代码如下:

class ThreadPool{
  using Task = std::function<void()>;


  public:    
      ThreadPool();
      ~ThreadPool();
      
      bool Init_ThreadPool();  //初始化线程       
      bool Add_task(const Task&t);  //添加任务 
      void End_threadpool();  //销毁线程            
      
      int thread_num;  //当前的线程数量
      int task_numble;   //任务队列 
    
  protected:
     void runtask(); 
     void Expand_thread();
     bool Destroy_thread();
     int id;
     
    private:
      
      int max_thread;//初始化线程数量 
      int max_queue;//初始化 任务队列数量 
    
      std::vector<std::thread> pool;// 线程池、任务队列 
      std::queue<Task>task ;
    
      std::mutex _mutex ;//互斥锁条件变量
      std::condition_variable condition; 
      
      bool stop;  //停止标志位 
    
      int key;  //目前正在执行任务的线程数量 
};

其中,~ThreadPool()的代码实现如下:

ThreadPool::~ThreadPool(){
  End_threadpool();
}
void ThreadPool::End_threadpool(){
  
  stop=true;  //停止读取任务 
  
  while(thread_run){}//等待线程还没有执行完的任务
  
  for(int i =0 ;i<pool.size();i++)  //销毁线程 
        pool[i].join() ;
   
  std::queue<Task> empty;
  swap(empty,task);//清空任务队列 
  pool.clear(); //清空线程池 
  
}

至此,线程池类封装就实现了,其余功能函数在这就不多讲了,有兴趣的同学可自行完成。

好了,今天关于C++实现线程池的话题,就讲到这里了,关于源代码的问题,我整理好后会放在公众号(小懵白生活小趣谈)后台里面。



Tags:线程池   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
原文链接: https://mp.weixin.qq.com/s/MTw7z6n_wk4y4CTmGkoRoA一切要从CPU说起你可能会有疑问,讲多线程为什么要从CPU说起呢?原因很简单,在这里没有那些时髦的概念,你可以更加清...【详细内容】
2021-08-13  Tags: 线程池  点击:(96)  评论:(0)  加入收藏
多线程并发是Java语言中非常重要的一块内容,同时,也是Java基础的一个难点。说它重要是因为多线程是日常开发中频繁用到的知识,说它难是因为多线程并发涉及到的知识点非常之多,想...【详细内容】
2021-07-12  Tags: 线程池  点击:(110)  评论:(0)  加入收藏
1. Dubbo简介及线程池策略Apache Dubbo 是一款高性能、轻量级的开源 Java 服务框架。提供了六大核心能力:面向接口代理的高性能RPC调用,智能容错和负载均衡,服务自动注册和发现...【详细内容】
2021-05-18  Tags: 线程池  点击:(202)  评论:(0)  加入收藏
在上一篇文章C++使用socket实现与微信小程序通信(下)中,小懵白就给大家简要地讲解了线程池的原理。 今天呢,小懵白就给大家继续讲解C++如何实现封装线程池类。第一步首先,我们需...【详细内容】
2021-05-14  Tags: 线程池  点击:(204)  评论:(0)  加入收藏
见字如面,我是威哥,一个从普通二本院校毕业,从未曾接触分布式、微服务、高并发到通过技术分享实现职场蜕变,成长为RocketMQ社区优秀布道师、大厂资深架构师,出版《RocketMQ技...【详细内容】
2021-03-31  Tags: 线程池  点击:(277)  评论:(0)  加入收藏
作者公众号:一角钱技术(org_yijiaoqian)前言线程池的具体实现有两种,分别是ThreadPoolExecutor 默认线程池和ScheduledThreadPoolExecutor 定时线程池,上一篇已经分析过ThreadPoo...【详细内容】
2020-12-22  Tags: 线程池  点击:(143)  评论:(0)  加入收藏
之前我们介绍了线程池的四种拒绝策略,了解了线程池参数的含义,那么今天我们来聊聊Java 中常见的几种线程池,以及在jdk7 加入的 ForkJoin 新型线程池 首先我们列出Java 中的...【详细内容】
2020-11-05  Tags: 线程池  点击:(91)  评论:(0)  加入收藏
前面几篇文章分析了线程的主要实现,今天来整体总结以下他们。总览图直接上总结的总览图,如下图: 如果看过前几篇文章应该基本能够看懂这张总结图,可能在单独的一篇文章里弄懂了...【详细内容】
2020-09-08  Tags: 线程池  点击:(63)  评论:(0)  加入收藏
大多数线程池实现都离不开锁的使用,如互斥量pthread_mutex*结合条件变量pthread_cond*。众所周知,锁的使用对于程序性能影响较大,虽然现有的pthread_mutex*在锁的申请与释放方...【详细内容】
2020-08-24  Tags: 线程池  点击:(87)  评论:(0)  加入收藏
作为 Java 程序员,无论是技术面试、项目研发或者是学习框架源码,不彻底掌握 Java 多线程的知识,做不到心中有数,干啥都没底气,尤其是技术深究时往往略显发憷。坐稳扶好,通过今天的...【详细内容】
2020-08-12  Tags: 线程池  点击:(48)  评论:(0)  加入收藏
▌简易百科推荐
一、简介很多时候我们都需要用到一些验证的方法,有时候需要用正则表达式校验数据时,往往需要到网上找很久,结果找到的还不是很符合自己想要的。所以我把自己整理的校验帮助类分...【详细内容】
2021-12-27  中年农码工    Tags:C#   点击:(1)  评论:(0)  加入收藏
引言在学习C语言或者其他编程语言的时候,我们编写的一个程序代码,基本都是在屏幕上打印出 hello world ,开始步入编程世(深)界(坑)的。C 语言版本的 hello world 代码:#include <std...【详细内容】
2021-12-21  一起学嵌入式    Tags:C 语言   点击:(10)  评论:(0)  加入收藏
读取SQLite数据库,就是读取一个路径\\192.168.100.**\position\db.sqlite下的文件<startup useLegacyV2RuntimeActivationPolicy="true"> <supportedRuntime version="v4.0"/...【详细内容】
2021-12-16  今朝我的奋斗    Tags:c#   点击:(21)  评论:(0)  加入收藏
什么是shellshell是c语言编写的程序,它在用户和操作系统之间架起了一座桥梁,用户可以通过这个桥梁访问操作系统内核服务。 它既是一种命令语言,同时也是一种程序设计语言,你可以...【详细内容】
2021-12-16  梦回故里归来    Tags:shell脚本   点击:(16)  评论:(0)  加入收藏
一、编程语言1.根据熟悉的语言,谈谈两种语言的区别?主要浅谈下C/C++和PHP语言的区别:1)PHP弱类型语言,一种脚本语言,对数据的类型不要求过多,较多的应用于Web应用开发,现在好多互...【详细内容】
2021-12-15  linux上的码农    Tags:c/c++   点击:(17)  评论:(0)  加入收藏
1.字符串数组+初始化char s1[]="array"; //字符数组char s2[6]="array"; //数组长度=字符串长度+1,因为字符串末尾会自动添&lsquo;\0&lsquo;printf("%s,%c\n",s1,s2[2]);...【详细内容】
2021-12-08  灯-灯灯    Tags:C语言   点击:(46)  评论:(0)  加入收藏
函数调用约定(Calling Convention),是一个重要的基础概念,用来规定调用者和被调用者是如何传递参数的,既调用者如何将参数按照什么样的规范传递给被调用者。在参数传递中,有两个很...【详细内容】
2021-11-30  小智雅汇    Tags:函数   点击:(19)  评论:(0)  加入收藏
一、问题提出问题:把m个苹果放入n个盘子中,允许有的盘子为空,共有多少种方法?注:5,1,1和1 5 1属同一种方法m,n均小于10二、算法分析设f(m,n) 为m个苹果,n个盘子的放法数目,则先对...【详细内容】
2021-11-17  C语言编程    Tags:C语言   点击:(46)  评论:(0)  加入收藏
一、为什么需要使用内存池在C/C++中我们通常使用malloc,free或new,delete来动态分配内存。一方面,因为这些函数涉及到了系统调用,所以频繁的调用必然会导致程序性能的损耗;另一...【详细内容】
2021-11-17  深度Linux    Tags:C++   点击:(37)  评论:(0)  加入收藏
OpenCV(Open Source Computer Vision Library)是一个(开源免费)发行的跨平台计算机视觉库,可以运行在Linux、Windows、Android、ios等操作系统上,它轻量级而且高效---由一系列...【详细内容】
2021-11-11  zls315    Tags:C#   点击:(50)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条