嵌入式八股,手撕线程池(C++)

打印 上一主题 下一主题

主题 1018|帖子 1018|积分 3058

线程池的主要目的是复用线程资源,减少线程创建和销毁的开销,同时提高程序的并发性能。
也就是说,我创建一个线程对象,他可以复用,线程池里有多个线程对象,有任务来了,我调用一个,用完了,我再放回去。
线程池优点

提高线程利用率
提高响应速率
便于同一管理线程对象
可以控制最大并发数
C++实现

  1. #include <iostream>
  2. #include <vector>
  3. #include <queue>
  4. #include <thread>
  5. #include <mutex>
  6. #include <condition_variable>
  7. #include <functional>
  8. #include <future>
  9. #include <memory>
  10. class threadpools
  11. {
  12. private:
  13.     /* data */
  14.     std::vector<std::thread> threads;// 一个线程数组,用来加入线程,线程池那个池
  15.     std::queue<std::function<void()>> task_q;//一个任务队列,用来发布任务,等待消费者来取
  16.     std::mutex mtx;//一个互斥量
  17.     std::condition_variable conditon;//条件变量,防止为空后继续取任务,起通知作用,管理多个线程
  18.     bool stop;//控制线程池的开始与结束
  19. public:
  20.     threadpools(int threadnums);
  21.     ~threadpools();
  22.     template <class F, class... Args>
  23.     void add_task(F&& f, Args&&... args);
  24.     std::atomic<int> task_count; // 添加一个任务计数器,防止主程序退出了,线程池里一个任务都没执行,我直接写公有了,不要学我
  25. };
  26. threadpools::threadpools(int threadnums):stop(false), task_count(0) {
  27.         //加线程到池中到最大数目
  28.         for (int i = 0; i < threadnums; i++)
  29.         {
  30.             threads.emplace_back([this]{//加入线程,push会拷贝加入的变量,这个不会
  31.                 while (1)
  32.                 {
  33.                     std::unique_lock<std::mutex> lock(mtx);//创建一个直接上锁
  34.                     conditon.wait(lock,[this]{
  35.                         return !this->task_q.empty() || stop;});//任务为空或线程池终止就等待结束,继续上锁,线程会卡在这里
  36.                     if(stop&&task_q.empty()){//一个双重检测,防止任务被其它线程取到了,导致任务列表为空从而报错,即竞态条件
  37.                         return;
  38.                     }
  39.                 //如果任务队列里有任务,我就取一个任务去完成
  40.                 std::function<void()> task(std::move(task_q.front()));
  41.                 task_q.pop();
  42.                 lock.unlock();//取到了解锁,让其它线程去取
  43.                 task();
  44.                 task_count--;
  45.                 }
  46.                
  47.             });
  48.         }
  49.         
  50.     }
  51. threadpools::~threadpools()
  52. {
  53.     std::unique_lock<std::mutex> lock(mtx);
  54.     stop = true;
  55.     conditon.notify_all();//通知所有线程工作,取完所有任务
  56.     for (auto& t:threads)
  57.     {
  58.         t.join();
  59.     }
  60.    
  61. }
  62. //用函数模版写一个加入任务的函数
  63. template <class F, class... Args>
  64. void threadpools::add_task(F &&f, Args&&... args){
  65.     std::function<void()> task = std::bind(std::forward<F>(f), std::forward<Args>(args)...);
  66.     std::unique_lock<std::mutex> lock(mtx);
  67.     task_q.emplace(std::move(task));
  68.     task_count++; // 添加任务,计数器加一
  69.     conditon.notify_one();
  70. }
  71. int main(){
  72.     threadpools pool(4);
  73.     for (int i = 0; i < 10; i++) {
  74.         std::cout << "加入任务: " << i << std::endl;
  75.         pool.add_task([i] {
  76.             std::this_thread::sleep_for(std::chrono::milliseconds(100));
  77.             printf("任务%d已完成\n", i);
  78.         });
  79.     }
  80.     // 等待线程池中的任务完成
  81.     while (pool.task_count > 0) {
  82.         std::this_thread::sleep_for(std::chrono::milliseconds(100));
  83.     }
  84.     return 0;
  85. }
复制代码
100行左右的CPP线程池实现,实现结果如下

我在执行add_task函数之前就输出了参加任务,按顺序参加,但输出任务实现的顺序是打乱的,符合线程池结果。
代码里还涉及到一些CPP11的新特性,比如完美转发,需要重点关注。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

王國慶

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表