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

聊聊Linux C下線程池的使用

系統(tǒng) Linux
線程池也是多線程的處理方式。是將“生產(chǎn)者”線程提出任務(wù)添加到“任務(wù)隊(duì)列”,然后一些線程自動(dòng)完成“任務(wù)隊(duì)列”上的任務(wù)。

[[379660]]

 線程池也是多線程的處理方式。是將“生產(chǎn)者”線程提出任務(wù)添加到“任務(wù)隊(duì)列”,然后一些線程自動(dòng)完成“任務(wù)隊(duì)列”上的任務(wù)。

多線程編程,創(chuàng)建一個(gè)線程,指定去完成某一個(gè)任務(wù),等待線程的退出。雖然能夠滿足編程需求,但是當(dāng)我們需要?jiǎng)?chuàng)建大量的線程的時(shí)候,在創(chuàng)建過(guò)程以及銷毀線程的過(guò)程中可能會(huì)消耗大量的CPU.增加很大開銷。如:文件夾的copy、WEB服務(wù)器的響應(yīng)。

線程池就是用來(lái)解決類似于這樣的一個(gè)問(wèn)題的,可以降低頻繁地創(chuàng)建和銷毀線程所帶來(lái)地開銷。

線程池技術(shù)思路:一般采用預(yù)創(chuàng)建線程技術(shù),也就是提前把需要用線程先創(chuàng)建一定數(shù)目。這些線程提前創(chuàng)建好了之后,“任務(wù)隊(duì)列”里面假設(shè)沒有任務(wù),那么就讓這些線程休眠,一旦有任務(wù),就喚醒線程去執(zhí)行任務(wù),任務(wù)執(zhí)行完了,也不需要去銷毀線程,直到當(dāng)你想退出或者是關(guān)機(jī)時(shí),這個(gè)時(shí)候,那么你調(diào)用銷毀線程池地函數(shù)去銷毀線程。

線程完成任務(wù)之后不會(huì)銷毀,而是自動(dòng)地執(zhí)行下一個(gè)任務(wù)。而且,當(dāng)任務(wù)有很多,你可以有函數(shù)接口去增加線程數(shù)量,當(dāng)任務(wù)較少時(shí),你可以有函數(shù)接口去銷毀部分線程。

如果,創(chuàng)建和銷毀線程的時(shí)間對(duì)比執(zhí)行任務(wù)的時(shí)間可以忽略不計(jì),那么我們?cè)谶@種情況下面也就沒有必要用線程池。

“任務(wù)隊(duì)列”是一個(gè)共享資源“互斥訪問(wèn)”


線程池本質(zhì)上也是一個(gè)數(shù)據(jù)結(jié)構(gòu),需要一個(gè)結(jié)構(gòu)體去描述它:

  1. struct pthread_pool //線程池的實(shí)現(xiàn) 
  2.  //一般會(huì)有如下成員 
  3.  
  4.  //互斥鎖,用來(lái)保護(hù)這個(gè)“任務(wù)隊(duì)列” 
  5.  pthread_mutex_t lock; //互斥鎖  
  6.   
  7.  //線程條件變量 表示“任務(wù)隊(duì)列”是否有任務(wù) 
  8.  pthread_cond_t cond; //條件變量 
  9.   
  10.  bool shutdown; //表示是否退出程序 bool:類型 false / true 
  11.  
  12.  //任務(wù)隊(duì)列(鏈表),指向第一個(gè)需要指向的任務(wù) 
  13.  //所有的線程都從任務(wù)鏈表中獲取任務(wù) "共享資源" 
  14.  struct task * task_list; 
  15.   
  16.  //線程池中有多個(gè)線程,每一個(gè)線程都有tid, 需要一個(gè)數(shù)組去保存tid 
  17.  pthread_t * tids; //malloc()  
  18.   
  19.  //線程池中正在服役的線程數(shù),當(dāng)前線程個(gè)數(shù) 
  20.  unsigned int active_threads; 
  21.   
  22.  //線程池任務(wù)隊(duì)列最大的任務(wù)數(shù)量 
  23.  unsigned int max_waiting_tasks; 
  24.   
  25.  //線程池任務(wù)隊(duì)列上當(dāng)前有多少個(gè)任務(wù) 
  26.  unsigned int cur_waiting_tasks; 
  27.   
  28.  //...... 
  29.  
  30. }; 
  31.  
  32. //任務(wù)隊(duì)列(鏈表)上面的任務(wù)結(jié)點(diǎn),只要能夠描述好一個(gè)任務(wù)就可以了, 
  33. //線程會(huì)不斷地任務(wù)隊(duì)列取任務(wù) 
  34. struct task  //任務(wù)結(jié)點(diǎn)  
  35.  // 1. 任務(wù)結(jié)點(diǎn)表示的任務(wù),“函數(shù)指針”指向任務(wù)要執(zhí)行的函數(shù)(cp_file) 
  36.  void*(* do_task)(void * arg); 
  37.   
  38.  //2. 指針,指向任務(wù)指向函數(shù)的參數(shù)(文件描述符) 
  39.  void * arg; 
  40.   
  41.  //3. 任務(wù)結(jié)點(diǎn)類型的指針,指向下一個(gè)任務(wù) 
  42.  struct task * next
  43. }; 

線程池框架代碼如下,功能自填:

操作線程池所需要的函數(shù)接口:pthread_pool.c 、pthread_pool.h

把“線程池”想象成一個(gè)外包公司,你需要去完成的就是操作線程池所提供的函數(shù)接口。

pthread_pool.c

  1. #include "pthread_pool.h" 
  2.  
  3. /* 
  4.  init_pool: 線程池初始化函數(shù),初始化指定的線程池中有thread_num個(gè)初始線程 
  5.  @pool:指針,指向您要初始化的那個(gè)線程池 
  6.  @threa_num: 您要初始化的線程池中開始的線程數(shù)量 
  7.  返回值:  
  8.   成功 0 
  9.   失敗 -1 
  10. */ 
  11.  
  12. int init_pool(pthread_pool * pool , unsigned int threa_num) 
  13.  //初始化線程池的結(jié)構(gòu)體 
  14.   
  15.  //初始化線程互斥鎖 
  16.  pthread_mutex_init(&pool->lock, NULL); 
  17.   
  18.  //初始化線程條件變量 
  19.  pthread_cond_init(&pool->cond, NULL); 
  20.  
  21.  pool->shutdown = false ;// 不退出 
  22.  
  23.  pool->task_list = (struct task*)malloc(sizeof(struct task)); 
  24.  
  25.  pool->tids = (pthread_t *)malloc(sizeof(pthread_t) * MAX_ACTIVE_THREADS); 
  26.  if(pool->task_list == NULL || pool->tids == NULL
  27.  { 
  28.   perror("malloc memery error"); 
  29.   return -1; 
  30.  } 
  31.  
  32.  pool->task_list->next = NULL
  33.  
  34.  //線程池中一開始初始化多少個(gè)線程來(lái)服役 
  35.  pool->active_threads = threa_num; 
  36.  
  37.  //表示線程池中最多有多少個(gè)任務(wù) 
  38.  pool->max_waiting_tasks = MAX_WAITING_TASKS; 
  39.  
  40.  //線程池中任務(wù)隊(duì)列當(dāng)前的任務(wù)數(shù)量 
  41.  pool->cur_waiting_tasks = 0; 
  42.  
  43.  //創(chuàng)建thread_num個(gè)線程,并且讓線程去執(zhí)行任務(wù)調(diào)配函數(shù), 
  44.  //記錄所有線程的tid 
  45.  int i = 0; 
  46.  for(i = 0; i < threa_num; i++) 
  47.  { 
  48.   int ret = pthread_create(&(pool->tids)[i], NULL, routine, (void*)pool); 
  49.   if(ret != 0) 
  50.   { 
  51.    perror("create thread error"); 
  52.    return -1; 
  53.   } 
  54.  
  55.   printf("[%lu]:[%s] ===> tids[%d]:[%lu]",pthread_self(), 
  56.    __FUNCTION__, i , pool->tids[i]); 
  57.  } 
  58.  
  59.  return 0; 
  60.  
  61. /* 
  62.  routine: 任務(wù)調(diào)配函數(shù)。 
  63.   所有線程開始都執(zhí)行此函數(shù),此函數(shù)會(huì)不斷的從線程池的任務(wù)隊(duì)列 
  64.   中取下任務(wù)結(jié)點(diǎn),去執(zhí)行。 
  65.    
  66.   任務(wù)結(jié)點(diǎn)中包含“函數(shù)指針” h "函數(shù)參數(shù)" 
  67. */ 
  68.  
  69. void * routine(void * arg) 
  70.  //arg表示你的線程池的指針 
  71.   
  72.  while() 
  73.  { 
  74.   //獲取線程互斥鎖,lock  
  75.    
  76.   //當(dāng)線程池沒有結(jié)束的時(shí)候,不斷地從線程池的任務(wù)隊(duì)列取下結(jié)點(diǎn) 
  77.   //去執(zhí)行。 
  78.    
  79.   //釋放線程互斥鎖,unlock 
  80.    
  81.   //釋放任務(wù)結(jié)點(diǎn) 
  82.  } 
  83.  
  84. /* 
  85.  destroy_pool: 銷毀線程池,銷毀前要保證所有的任務(wù)已經(jīng)完成 
  86. */ 
  87.  
  88. int destroy_pool(pthread_pool * pool) 
  89.  //釋放所有空間 等待任務(wù)執(zhí)行完畢(join)。 
  90.  //喚醒所有線程 
  91.  //利用join函數(shù)回收每一個(gè)線程資源。 
  92.  
  93. /* 
  94.  add_task:給任務(wù)隊(duì)列增加任務(wù), 把do_task指向的任務(wù)(函數(shù)指針)和 
  95.   arg指向的參數(shù)保存到一個(gè)任務(wù)結(jié)點(diǎn),添加到pool任務(wù)隊(duì)列中。 
  96.    
  97.  @pool : 您要添加任務(wù)的線程池 
  98.  @do_task : 您需要添加的任務(wù)(cp_file) 
  99.  @arg: 您要執(zhí)行的任務(wù)的參數(shù)(文件描述符) 
  100. */ 
  101.  
  102. int add_task(pthread_pool *pool,void*(* do_task)(void * arg), void*arg) 
  103.  //把第二個(gè)參數(shù)和第三個(gè)參數(shù)封裝成struct task  
  104.   
  105.  //再把它添加到 pool->task 任務(wù)隊(duì)列中去 
  106.   
  107.  //注意任務(wù)隊(duì)列是一個(gè)共享資源 
  108.   
  109.  //假如任務(wù)后要喚醒等待的線程。 
  110.  
  111. //如果任務(wù)多的時(shí)候,往線程池中添加線程  pthread_create 
  112. int add_threads(pthread_pool * pool, unsigned int num); 
  113.  //新創(chuàng)建num個(gè)線程,讓每一個(gè)線程去執(zhí)行線程調(diào)配函數(shù) 
  114.   
  115.  //將每一個(gè)新創(chuàng)建的線程tid,添加到pool-> tids  
  116.  
  117. //如果任務(wù)少的時(shí)候,減少線程池中線程的數(shù)量 pthread_cancel join 
  118. int remove_threads(pthread_pool * pool, unsigned int num) 
  119.  //用pthread_cancel取消num個(gè)線程  
  120.  //利用pthread_join函數(shù)去回收資源。 

pthread_pool.h

  1. #ifndef __PTHREAD_POOL_H__ 
  2. #define __PTHREAD_POOL_H__ 
  3.  
  4. //表示線程池中最多有多少個(gè)線程 
  5. #define MAX_ACTIVE_THREADS 20 
  6.  
  7. //表示線程池中最多有多少個(gè)任務(wù) 
  8. #define MAX_WAITING_TASKS 1024 
  9.  
  10. //任務(wù)隊(duì)列(鏈表)上面的任務(wù)結(jié)點(diǎn),只要能夠描述好一個(gè)任務(wù)就可以了, 
  11. //線程會(huì)不斷地任務(wù)隊(duì)列取任務(wù) 
  12. struct task  //任務(wù)結(jié)點(diǎn)  
  13.  // 1. 任務(wù)結(jié)點(diǎn)表示的任務(wù),“函數(shù)指針”指向任務(wù)要執(zhí)行的函數(shù)(cp_file) 
  14.  void*(* do_task)(void * arg); 
  15.   
  16.  //2. 指針,指向任務(wù)指向函數(shù)的參數(shù)(文件描述符) 
  17.  void * arg; 
  18.   
  19.  //3. 任務(wù)結(jié)點(diǎn)類型的指針,指向下一個(gè)任務(wù) 
  20.  struct task * next
  21. }; 
  22.  
  23. struct pthread_pool //線程池的實(shí)現(xiàn) 
  24.  //一般會(huì)有如下成員 
  25.  
  26.  //互斥鎖,用來(lái)保護(hù)這個(gè)“任務(wù)隊(duì)列” 
  27.  pthread_mutex_t lock; //互斥鎖  
  28.   
  29.  //線程條件變量 表示“任務(wù)隊(duì)列”是否有任務(wù) 
  30.  pthread_cond_t cond; //條件變量 
  31.   
  32.  bool shutdown; //表示是否退出程序 bool:類型 false / true 
  33.  
  34.  //任務(wù)隊(duì)列(鏈表),指向第一個(gè)需要指向的任務(wù) 
  35.  //所有的線程都從任務(wù)鏈表中獲取任務(wù) "共享資源" 
  36.  struct task * task_list; 
  37.   
  38.  //線程池中有多個(gè)線程,每一個(gè)線程都有tid, 需要一個(gè)數(shù)組去保存tid 
  39.  pthread_t * tids; //malloc()  
  40.   
  41.  //線程池中正在服役的線程數(shù),當(dāng)前線程個(gè)數(shù) 
  42.  unsigned int active_threads; 
  43.   
  44.  //線程池任務(wù)隊(duì)列最大的任務(wù)數(shù)量 
  45.  unsigned int max_waiting_tasks; 
  46.   
  47.  //線程池任務(wù)隊(duì)列上當(dāng)前有多少個(gè)任務(wù) 
  48.  unsigned int cur_waiting_tasks; 
  49.   
  50.  //...... 
  51.  
  52. }; 
  53.  
  54. /* 
  55.  init_pool: 線程池初始化函數(shù),初始化指定的線程池中有thread_num 
  56.   個(gè)初始線程 
  57.  @pool:指針,指向您要初始化的那個(gè)線程池 
  58.  @threa_num: 您要初始化的線程池中開始的線程數(shù)量 
  59.  返回值:  
  60.   成功 0 
  61.   失敗 -1 
  62. */ 
  63.  
  64. int init_pool(pthread_pool * pool , unsigned int threa_num); 
  65.  
  66. /* 
  67.  routine: 任務(wù)調(diào)配函數(shù)。 
  68.   所有線程開始都執(zhí)行此函數(shù),此函數(shù)會(huì)不斷的從線程池的任務(wù)隊(duì)列 
  69.   中取下任務(wù)結(jié)點(diǎn),去執(zhí)行。 
  70.    
  71.   任務(wù)結(jié)點(diǎn)中包含“函數(shù)指針” h "函數(shù)參數(shù)" 
  72. */ 
  73.  
  74. void * routine(void * arg); 
  75.  
  76. /* 
  77.  destroy_pool: 銷毀線程池,銷毀前要保證所有的任務(wù)已經(jīng)完成 
  78. */ 
  79.  
  80. int destroy_pool(pthread_pool * pool); 
  81.  
  82. /* 
  83.  add_task:給任務(wù)隊(duì)列增加任務(wù), 把do_task指向的任務(wù)(函數(shù)指針)和 
  84.   arg指向的參數(shù)保存到一個(gè)任務(wù)結(jié)點(diǎn),添加到pool任務(wù)隊(duì)列中。 
  85.    
  86.  @pool : 您要添加任務(wù)的線程池 
  87.  @do_task : 您需要添加的任務(wù)(cp_file) 
  88.  @arg: 您要執(zhí)行的任務(wù)的參數(shù)(文件描述符) 
  89. */ 
  90.  
  91. int add_task(pthread_pool *pool,void*(* do_task)(void * arg), void*arg); 
  92.  
  93. //如果任務(wù)多的時(shí)候,往線程池中添加線程  pthread_create 
  94. int add_threads(pthread_pool * pool, unsigned int num); 
  95.  
  96.  
  97. //如果任務(wù)少的時(shí)候,減少線程池中線程的數(shù)量 pthread_cancel join 
  98. int remove_threads(pthread_pool * pool, unsigned int num); 
  99.  
  100. #endif 

 

責(zé)任編輯:姜華 來(lái)源: 嵌入式Linux系統(tǒng)開發(fā)
相關(guān)推薦

2023-11-29 16:38:12

線程池阻塞隊(duì)列開發(fā)

2024-05-21 11:09:17

2022-08-29 09:06:43

hippo4j動(dòng)態(tài)線程池

2020-06-11 11:36:49

線程池Java場(chǎng)景

2024-06-04 07:52:04

2011-08-09 15:25:14

線程池數(shù)據(jù)庫(kù)連接池

2024-10-21 16:59:37

C#編程多線程

2021-09-11 15:26:23

Java多線程線程池

2023-07-11 08:34:25

參數(shù)流程類型

2022-09-06 08:31:09

線程池工具系統(tǒng)

2022-02-07 11:55:00

linux進(jìn)程線程

2025-02-28 08:46:24

框架微服務(wù)架構(gòu)

2012-02-29 13:26:20

Java

2024-12-10 00:00:25

2024-03-11 08:15:43

參數(shù)線程池方法

2022-03-09 07:35:24

線程池線程參數(shù)

2020-03-05 15:34:16

線程池C語(yǔ)言局域網(wǎng)

2024-10-06 14:37:52

2021-12-28 15:10:01

線程池C語(yǔ)言編程語(yǔ)言

2024-12-27 09:08:25

點(diǎn)贊
收藏

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