自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

掰開揉碎了教你設(shè)計(jì)線程池!還不來學(xué)?

開發(fā) 前端
池的概念是一種非常常見的空間換時(shí)間的概念,除了有線程池之外還有進(jìn)程池、內(nèi)存池等等。其實(shí)他們的思想都是一樣的就是我先申請一批資源出來,然后就隨用隨拿,不用再放回來。聽到這兒是不是有種云計(jì)算的思想了,他們道理都是一樣的。

[[438359]]

大家好,我是作者小杰,我在學(xué)習(xí)線程池的時(shí)候也曾查閱過各種資料,但是感覺大佬寫的很好但是寫的不夠詳細(xì),寫的詳細(xì)的設(shè)計(jì)思路又很簡單,所以我的出發(fā)點(diǎn)就是讓讀者可以清晰明確的看懂整個設(shè)計(jì)思想和設(shè)計(jì)過程,可以舉一反三,在今后內(nèi)存池等方面也可以游刃有余的設(shè)計(jì)出來!好了,正文開始~

線程池設(shè)計(jì)思路

線程池是什么

我們先來打個比方,線程池就好像一個工具箱,我們每次需要擰螺絲的時(shí)候都要從工具箱里面取出一個螺絲刀來,有時(shí)候需要取出一個來擰,有時(shí)候螺絲多的時(shí)候需要多個人取出多個來擰,擰完自己的螺絲那么就會把螺絲刀再放回去,然后別人下次用的時(shí)候再取出來用。也許我的例子不是太完美,但是我想我已經(jīng)基本闡述清楚了線程池。說白了線程池就是相當(dāng)于提前申請了一些資源也就是線程,需要的時(shí)候就從線程池中取出線程來處理一些事情,處理完畢之后再把線程放回去。

線程池介紹

為什么需要線程池

我們來思考一個問題,為什么需要線程池呢?假如沒有線程池的話我們每次調(diào)用線程是什么樣子的?顯然首先是先創(chuàng)建一個線程,然后再把任務(wù)交給這個線程,最后再把這個線程銷毀掉。那么如果我們改用線程池的話,我們在程序運(yùn)行的時(shí)候就會首先創(chuàng)建一批線程,然后交給線程池來管理。有需要的時(shí)候我們把線程拿出去處理任務(wù),不需要的時(shí)候我們再回收到線程池中,這樣是不是就避免了每次都需要創(chuàng)建和銷毀線程這種消耗時(shí)間的操作。有人會說你使用線程池一開始就消耗了一些內(nèi)存,之后一直不釋放這些內(nèi)存,這樣豈不是有點(diǎn)浪費(fèi)。其實(shí)這是類似于空間換時(shí)間的概念,我們確實(shí)多占用了一點(diǎn)內(nèi)存但是這些內(nèi)存和我們珍惜出來的這些時(shí)間相比,是非常劃算的。

池的概念是一種非常常見的空間換時(shí)間的概念,除了有線程池之外還有進(jìn)程池、內(nèi)存池等等。其實(shí)他們的思想都是一樣的就是我先申請一批資源出來,然后就隨用隨拿,不用再放回來。聽到這兒是不是有種云計(jì)算的思想了,他們道理都是一樣的。

如何設(shè)計(jì)線程池

現(xiàn)在硬核的知識要開始了,請坐穩(wěn)扶好、抓緊扶手~

二話不說,先上圖看看,我們要設(shè)計(jì)的線程池長什么樣子!

線程池的設(shè)計(jì)

設(shè)計(jì)思路

我們需要一個線程池類,那么線程池類中都需要哪些東西呢?我們庖丁解牛來看一看

  • 我們需要存放我們創(chuàng)建好的線程,因此我們需要一個容器專門放線程
  • 需要一個容器來存放我們的任務(wù),每次把任務(wù)放到這個容器里面
  • 由于是多線程的讀取任務(wù),所以必不可少的我們需要鎖,每次讀取任務(wù)需要加鎖和解鎖
  • 我們需要判斷什么時(shí)候終止,因此還需要一個判斷終止的變量

為了避免輪詢的判斷任務(wù)集裝箱里面是不是空的,這樣效率太低了,因此我們這里采用條件變量

這里來說明一下什么是條件變量。條件變量是并發(fā)編程中的一種同步機(jī)制,條件變量使得線程能夠阻塞到等待某個條件發(fā)生后,再繼續(xù)執(zhí)行,期間還會把之前拿到的鎖先釋放掉,不影響其它人拿這把鎖。因此條件變量十分強(qiáng)大而高效。(條件變量和鎖將會在我多線程文章中詳細(xì)講解,這里不是重點(diǎn),所以不再展開細(xì)講)

接下來我們來研究一下線程池中需要有哪些操作呢?

  • 將任務(wù)添加到線程池中的操作,并且這時(shí)應(yīng)該通知線程可以來取任務(wù)來執(zhí)行了
  • 一個循環(huán)操作,不斷地等待任務(wù)集裝箱里面有數(shù)據(jù)來執(zhí)行,也就是初始化完畢后需要做的事情
  • 通過改變終止變量來讓上面循環(huán)停止的操作

好了,到此已經(jīng)詳細(xì)的把設(shè)計(jì)思路寫清楚了,接下來該看具體的實(shí)現(xiàn)了

線程池的實(shí)現(xiàn)

接下來先來看一看線程池類是怎么實(shí)現(xiàn)的,注釋已經(jīng)很詳細(xì)了,就不多說了直接上代碼。

  1. class CThreadMangerPool 
  2. public
  3.  CThreadMangerPool(void):is_runing(false){}; 
  4.  bool init(int threadnum);//初始化函數(shù) 
  5.  ~CThreadMangerPool(void); 
  6.  void Run(void);  //執(zhí)行函數(shù) 
  7.  void stop(void); //用來終止循環(huán)的函數(shù) 
  8.  void addTask(ThreadTask* task);//向任務(wù)集裝箱中添加任務(wù)的函數(shù) 
  9. private: 
  10.  bool CreateThreads(int threadnum = 5); 
  11.  std::vector<std::shared_ptr<std::thread>> threadsPool;    //線程集裝箱,用來存放線程 
  12.  std::list<std::shared_ptr<ThreadTask>>    threadTaskList; //任務(wù)集裝箱,用來存放線程執(zhí)行的任務(wù) 
  13.  std::condition_variable       threadPool_cv;  //條件變量 
  14.  std::mutex          threadMutex;   //互斥鎖 
  15.  //std::vector<std::shared_ptr<CTcpClient>>  tcpClients; 
  16.  bool is_runing; //終止變量 
  17. }; 

我們來幾個重點(diǎn)的函數(shù)實(shí)現(xiàn)~

在Run函數(shù)中,我們設(shè)計(jì)了一個循環(huán),不斷地執(zhí)行等待并取出任務(wù)執(zhí)行,如果沒有的任務(wù)可以執(zhí)行的話就睡眠等待(用之前提到的條件變量來實(shí)現(xiàn))

注意這里使用了一個手法,我們用while來判斷任務(wù)集裝箱中的數(shù)據(jù)是不是空的,是因?yàn)轭愃朴谶M(jìn)程的驚群現(xiàn)象,這里出現(xiàn)條件變量的虛假喚醒。(在這里并不是重點(diǎn)就不展開講了,會在我文章的多線程處詳細(xì)講解)

  1. void CThreadMangerPool::Run(){ 
  2.  std::shared_ptr<ThreadTask> task;  
  3.  while(true){ //處在循環(huán)中 
  4.  
  5.   std::unique_lock<std::mutex> guard(threadMutex);//利用RALL來管理鎖,不用手動釋放 
  6.  
  7.   while(threadTaskList.empty()){ // 這里防止條件變量的虛假喚醒,所以不用if判斷 
  8.    if (!is_runing) 
  9.     break; 
  10.    threadPool_cv.wait(guard); //條件變量的使用 
  11.   } 
  12.   if (!is_runing)  //同上 都是判斷如果未啟動或者調(diào)用了stop函數(shù)都會退出循環(huán) 
  13.    break; 
  14.  
  15.   task = threadTaskList.front(); //取出任務(wù) 
  16.   threadTaskList.pop_front(); //把任務(wù)從容器中拿走 
  17.  
  18.   if (task == NULL
  19.    continue
  20.  
  21.   task->DoIt(); //執(zhí)行任務(wù)處理函數(shù) 
  22.   task.reset(); //重置指針 
  23.  } 
  24.  

接下來看看增加任務(wù)的函數(shù)是怎么實(shí)現(xiàn)的

  1. void CThreadMangerPool::addTask(ThreadTask* task){ 
  2.  std::shared_ptr<ThreadTask> ptr; //創(chuàng)建一個指向任務(wù)的智能指針 
  3.  ptr.reset(task); 
  4.  { 
  5.   std::lock_guard<std::mutex> guard(threadMutex);  //同樣是用RALL來管理鎖,免去手動釋放 
  6.   threadTaskList.push_back(ptr); //往任務(wù)集裝箱中添加任務(wù) 
  7.  } 
  8.  threadPool_cv.notify_all(); //通知線程可以執(zhí)行了,就是喚醒剛才在條件變量處睡眠的條件 
  9.   

好了,重點(diǎn)函數(shù)已經(jīng)看完了,其他的輕松就可以實(shí)現(xiàn)包括初始化函數(shù),終止函數(shù)等等

 

完結(jié)撒花~

 

責(zé)任編輯:武曉燕 來源: 暢游碼海
相關(guān)推薦

2025-04-09 08:21:10

2022-01-13 06:59:40

廣告Cookie項(xiàng)目

2023-08-09 09:03:49

CPU密集型運(yùn)算

2013-05-23 15:59:00

線程池

2020-10-19 10:01:12

Nodejs線程池設(shè)計(jì)

2024-11-06 09:39:52

2016-02-24 11:49:00

2021-06-08 09:49:01

協(xié)程池Golang設(shè)計(jì)

2020-07-23 14:39:28

系統(tǒng)權(quán)限設(shè)計(jì)

2023-05-19 08:01:24

Key消費(fèi)場景

2024-07-15 08:20:24

2023-12-06 08:28:44

禮物系統(tǒng)用例圖

2024-10-16 10:11:52

2022-09-19 18:32:22

函數(shù)編程語言

2012-05-15 02:18:31

Java線程池

2020-12-10 08:24:40

線程池線程方法

2023-10-13 08:20:02

Spring線程池id

2013-08-12 11:18:00

2023-06-07 13:49:00

多線程編程C#

2025-01-09 11:24:59

線程池美團(tuán)動態(tài)配置中心
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號