聊聊Linux C下线程池的使用

系统 Linux
线程池也是多线程的处理方式。是将“生产者”线程提出任务添加到“任务队列”,然后一些线程自动完成“任务队列”上的任务。

[[379660]]

 线程池也是多线程的处理方式。是将“生产者”线程提出任务添加到“任务队列”,然后一些线程自动完成“任务队列”上的任务。

多线程编程,创建一个线程,指定去完成某一个任务,等待线程的退出。虽然能够满足编程需求,但是当我们需要创建大量的线程的时候,在创建过程以及销毁线程的过程中可能会消耗大量的CPU.增加很大开销。如:文件夹的copy、WEB服务器的响应。

线程池就是用来解决类似于这样的一个问题的,可以降低频繁地创建和销毁线程所带来地开销。

线程池技术思路:一般采用预创建线程技术,也就是提前把需要用线程先创建一定数目。这些线程提前创建好了之后,“任务队列”里面假设没有任务,那么就让这些线程休眠,一旦有任务,就唤醒线程去执行任务,任务执行完了,也不需要去销毁线程,直到当你想退出或者是关机时,这个时候,那么你调用销毁线程池地函数去销毁线程。

线程完成任务之后不会销毁,而是自动地执行下一个任务。而且,当任务有很多,你可以有函数接口去增加线程数量,当任务较少时,你可以有函数接口去销毁部分线程。

如果,创建和销毁线程的时间对比执行任务的时间可以忽略不计,那么我们在这种情况下面也就没有必要用线程池。

“任务队列”是一个共享资源“互斥访问”


线程池本质上也是一个数据结构,需要一个结构体去描述它:

  1. struct pthread_pool //线程池的实现 
  2.  //一般会有如下成员 
  3.  
  4.  //互斥锁,用来保护这个“任务队列” 
  5.  pthread_mutex_t lock; //互斥锁  
  6.   
  7.  //线程条件变量 表示“任务队列”是否有任务 
  8.  pthread_cond_t cond; //条件变量 
  9.   
  10.  bool shutdown; //表示是否退出程序 bool:类型 false / true 
  11.  
  12.  //任务队列(链表),指向第一个需要指向的任务 
  13.  //所有的线程都从任务链表中获取任务 "共享资源" 
  14.  struct task * task_list; 
  15.   
  16.  //线程池中有多个线程,每一个线程都有tid, 需要一个数组去保存tid 
  17.  pthread_t * tids; //malloc()  
  18.   
  19.  //线程池中正在服役的线程数,当前线程个数 
  20.  unsigned int active_threads; 
  21.   
  22.  //线程池任务队列最大的任务数量 
  23.  unsigned int max_waiting_tasks; 
  24.   
  25.  //线程池任务队列上当前有多少个任务 
  26.  unsigned int cur_waiting_tasks; 
  27.   
  28.  //...... 
  29.  
  30. }; 
  31.  
  32. //任务队列(链表)上面的任务结点,只要能够描述好一个任务就可以了, 
  33. //线程会不断地任务队列取任务 
  34. struct task  //任务结点  
  35.  // 1. 任务结点表示的任务,“函数指针”指向任务要执行的函数(cp_file) 
  36.  void*(* do_task)(void * arg); 
  37.   
  38.  //2. 指针,指向任务指向函数的参数(文件描述符) 
  39.  void * arg; 
  40.   
  41.  //3. 任务结点类型的指针,指向下一个任务 
  42.  struct task * next
  43. }; 

线程池框架代码如下,功能自填:

操作线程池所需要的函数接口:pthread_pool.c 、pthread_pool.h

把“线程池”想象成一个外包公司,你需要去完成的就是操作线程池所提供的函数接口。

pthread_pool.c

  1. #include "pthread_pool.h" 
  2.  
  3. /* 
  4.  init_pool: 线程池初始化函数,初始化指定的线程池中有thread_num个初始线程 
  5.  @pool:指针,指向您要初始化的那个线程池 
  6.  @threa_num: 您要初始化的线程池中开始的线程数量 
  7.  返回值:  
  8.   成功 0 
  9.   失败 -1 
  10. */ 
  11.  
  12. int init_pool(pthread_pool * pool , unsigned int threa_num) 
  13.  //初始化线程池的结构体 
  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.  //线程池中一开始初始化多少个线程来服役 
  35.  pool->active_threads = threa_num; 
  36.  
  37.  //表示线程池中最多有多少个任务 
  38.  pool->max_waiting_tasks = MAX_WAITING_TASKS; 
  39.  
  40.  //线程池中任务队列当前的任务数量 
  41.  pool->cur_waiting_tasks = 0; 
  42.  
  43.  //创建thread_num个线程,并且让线程去执行任务调配函数, 
  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: 任务调配函数。 
  63.   所有线程开始都执行此函数,此函数会不断的从线程池的任务队列 
  64.   中取下任务结点,去执行。 
  65.    
  66.   任务结点中包含“函数指针” h "函数参数" 
  67. */ 
  68.  
  69. void * routine(void * arg) 
  70.  //arg表示你的线程池的指针 
  71.   
  72.  while() 
  73.  { 
  74.   //获取线程互斥锁,lock  
  75.    
  76.   //当线程池没有结束的时候,不断地从线程池的任务队列取下结点 
  77.   //去执行。 
  78.    
  79.   //释放线程互斥锁,unlock 
  80.    
  81.   //释放任务结点 
  82.  } 
  83.  
  84. /* 
  85.  destroy_pool: 销毁线程池,销毁前要保证所有的任务已经完成 
  86. */ 
  87.  
  88. int destroy_pool(pthread_pool * pool) 
  89.  //释放所有空间 等待任务执行完毕(join)。 
  90.  //唤醒所有线程 
  91.  //利用join函数回收每一个线程资源。 
  92.  
  93. /* 
  94.  add_task:给任务队列增加任务, 把do_task指向的任务(函数指针)和 
  95.   arg指向的参数保存到一个任务结点,添加到pool任务队列中。 
  96.    
  97.  @pool : 您要添加任务的线程池 
  98.  @do_task : 您需要添加的任务(cp_file) 
  99.  @arg: 您要执行的任务的参数(文件描述符) 
  100. */ 
  101.  
  102. int add_task(pthread_pool *pool,void*(* do_task)(void * arg), void*arg) 
  103.  //把第二个参数和第三个参数封装成struct task  
  104.   
  105.  //再把它添加到 pool->task 任务队列中去 
  106.   
  107.  //注意任务队列是一个共享资源 
  108.   
  109.  //假如任务后要唤醒等待的线程。 
  110.  
  111. //如果任务多的时候,往线程池中添加线程  pthread_create 
  112. int add_threads(pthread_pool * pool, unsigned int num); 
  113.  //新创建num个线程,让每一个线程去执行线程调配函数 
  114.   
  115.  //将每一个新创建的线程tid,添加到pool-> tids  
  116.  
  117. //如果任务少的时候,减少线程池中线程的数量 pthread_cancel join 
  118. int remove_threads(pthread_pool * pool, unsigned int num) 
  119.  //用pthread_cancel取消num个线程  
  120.  //利用pthread_join函数去回收资源。 

pthread_pool.h

  1. #ifndef __PTHREAD_POOL_H__ 
  2. #define __PTHREAD_POOL_H__ 
  3.  
  4. //表示线程池中最多有多少个线程 
  5. #define MAX_ACTIVE_THREADS 20 
  6.  
  7. //表示线程池中最多有多少个任务 
  8. #define MAX_WAITING_TASKS 1024 
  9.  
  10. //任务队列(链表)上面的任务结点,只要能够描述好一个任务就可以了, 
  11. //线程会不断地任务队列取任务 
  12. struct task  //任务结点  
  13.  // 1. 任务结点表示的任务,“函数指针”指向任务要执行的函数(cp_file) 
  14.  void*(* do_task)(void * arg); 
  15.   
  16.  //2. 指针,指向任务指向函数的参数(文件描述符) 
  17.  void * arg; 
  18.   
  19.  //3. 任务结点类型的指针,指向下一个任务 
  20.  struct task * next
  21. }; 
  22.  
  23. struct pthread_pool //线程池的实现 
  24.  //一般会有如下成员 
  25.  
  26.  //互斥锁,用来保护这个“任务队列” 
  27.  pthread_mutex_t lock; //互斥锁  
  28.   
  29.  //线程条件变量 表示“任务队列”是否有任务 
  30.  pthread_cond_t cond; //条件变量 
  31.   
  32.  bool shutdown; //表示是否退出程序 bool:类型 false / true 
  33.  
  34.  //任务队列(链表),指向第一个需要指向的任务 
  35.  //所有的线程都从任务链表中获取任务 "共享资源" 
  36.  struct task * task_list; 
  37.   
  38.  //线程池中有多个线程,每一个线程都有tid, 需要一个数组去保存tid 
  39.  pthread_t * tids; //malloc()  
  40.   
  41.  //线程池中正在服役的线程数,当前线程个数 
  42.  unsigned int active_threads; 
  43.   
  44.  //线程池任务队列最大的任务数量 
  45.  unsigned int max_waiting_tasks; 
  46.   
  47.  //线程池任务队列上当前有多少个任务 
  48.  unsigned int cur_waiting_tasks; 
  49.   
  50.  //...... 
  51.  
  52. }; 
  53.  
  54. /* 
  55.  init_pool: 线程池初始化函数,初始化指定的线程池中有thread_num 
  56.   个初始线程 
  57.  @pool:指针,指向您要初始化的那个线程池 
  58.  @threa_num: 您要初始化的线程池中开始的线程数量 
  59.  返回值:  
  60.   成功 0 
  61.   失败 -1 
  62. */ 
  63.  
  64. int init_pool(pthread_pool * pool , unsigned int threa_num); 
  65.  
  66. /* 
  67.  routine: 任务调配函数。 
  68.   所有线程开始都执行此函数,此函数会不断的从线程池的任务队列 
  69.   中取下任务结点,去执行。 
  70.    
  71.   任务结点中包含“函数指针” h "函数参数" 
  72. */ 
  73.  
  74. void * routine(void * arg); 
  75.  
  76. /* 
  77.  destroy_pool: 销毁线程池,销毁前要保证所有的任务已经完成 
  78. */ 
  79.  
  80. int destroy_pool(pthread_pool * pool); 
  81.  
  82. /* 
  83.  add_task:给任务队列增加任务, 把do_task指向的任务(函数指针)和 
  84.   arg指向的参数保存到一个任务结点,添加到pool任务队列中。 
  85.    
  86.  @pool : 您要添加任务的线程池 
  87.  @do_task : 您需要添加的任务(cp_file) 
  88.  @arg: 您要执行的任务的参数(文件描述符) 
  89. */ 
  90.  
  91. int add_task(pthread_pool *pool,void*(* do_task)(void * arg), void*arg); 
  92.  
  93. //如果任务多的时候,往线程池中添加线程  pthread_create 
  94. int add_threads(pthread_pool * pool, unsigned int num); 
  95.  
  96.  
  97. //如果任务少的时候,减少线程池中线程的数量 pthread_cancel join 
  98. int remove_threads(pthread_pool * pool, unsigned int num); 
  99.  
  100. #endif 

 

责任编辑:姜华 来源: 嵌入式Linux系统开发
相关推荐

2023-11-29 16:38:12

线程池阻塞队列开发

2024-05-21 11:09:17

2022-08-29 09:06:43

hippo4j动态线程池

2020-06-11 11:36:49

线程池Java场景

2024-10-21 16:59:37

C#编程多线程

2011-08-09 15:25:14

线程池数据库连接池

2024-06-04 07:52:04

2023-07-11 08:34:25

参数流程类型

2021-09-11 15:26:23

Java多线程线程池

2022-09-06 08:31:09

线程池工具系统

2022-02-07 11:55:00

linux进程线程

2024-12-10 00:00:25

2012-02-29 13:26:20

Java

2024-03-11 08:15:43

参数线程池方法

2022-03-09 07:35:24

线程池线程参数

2020-03-05 15:34:16

线程池C语言局域网

2024-10-06 14:37:52

2024-12-27 09:08:25

2021-12-28 15:10:01

线程池C语言编程语言

2022-10-12 09:01:52

Linux内核线程
点赞
收藏

51CTO技术栈公众号