转自:http://blog.csdn.net/lmh12506/article/details/7753952

前段时间在github上开了个库,准备实现自己的线程池的,因为换工作的事,一直也没有实现,参考这篇文章准备着手实现一下。

什么时候需要创建线程池呢?简单的说,如果一个应用需要频繁的创建和销毁线程,而任务执行的时间又非常短,这样线程创建和销毁的带来的开销就不容忽视,这时也是线程池该出场的机会了。如果线程创建和销毁时间相比任务执行时间可以忽略不计,则没有必要使用线程池了。

下面是Linux系统下用C语言创建的一个线程池。线程池会维护一个任务链表(每个CThread_worker结构就是一个任务)。

pool_init()函数预先创建好max_thread_num个线程,每个线程执thread_routine ()函数。该函数中

  1. while (pool->cur_queue_size == 0)
  2. {
  3. pthread_cond_wait (&(pool->queue_ready),&(pool->queue_lock));
  4. }

表示如果任务链表中没有任务,则该线程出于阻塞等待状态。否则从队列中取出任务并执行。
   
   pool_add_worker()函数向线程池的任务链表中加入一个任务,加入后通过调用pthread_cond_signal (&(pool->queue_ready))唤醒一个出于阻塞状态的线程(如果有的话)。
   
   pool_destroy ()函数用于销毁线程池,线程池任务链表中的任务不会再被执行,但是正在运行的线程会一直把任务运行完后再退出。

下面贴出完整代码

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <sys/types.h>
  5. #include <pthread.h>
  6. #include <assert.h>
  7. /*
  8. *线程池里所有运行和等待的任务都是一个CThread_worker
  9. *由于所有任务都在链表里,所以是一个链表结构
  10. */
  11. typedef struct worker
  12. {
  13. /*回调函数,任务运行时会调用此函数,注意也可声明成其它形式*/
  14. void *(*process) (void *arg);
  15. void *arg;/*回调函数的参数*/
  16. struct worker *next;
  17. } CThread_worker;
  18. /*线程池结构*/
  19. typedef struct
  20. {
  21. pthread_mutex_t queue_lock;
  22. pthread_cond_t queue_ready;
  23. /*链表结构,线程池中所有等待任务*/
  24. CThread_worker *queue_head;
  25. /*是否销毁线程池*/
  26. int shutdown;
  27. pthread_t *threadid;
  28. /*线程池中允许的活动线程数目*/
  29. int max_thread_num;
  30. /*当前等待队列的任务数目*/
  31. int cur_queue_size;
  32. } CThread_pool;
  33. int pool_add_worker (void *(*process) (void *arg), void *arg);
  34. void *thread_routine (void *arg);
  35. static CThread_pool *pool = NULL;
  36. void
  37. pool_init (int max_thread_num)
  38. {
  39. pool = (CThread_pool *) malloc (sizeof (CThread_pool));
  40. pthread_mutex_init (&(pool->queue_lock), NULL);
  41. pthread_cond_init (&(pool->queue_ready), NULL);
  42. pool->queue_head = NULL;
  43. pool->max_thread_num = max_thread_num;
  44. pool->cur_queue_size = 0;
  45. pool->shutdown = 0;
  46. pool->threadid =
  47. (pthread_t *) malloc (max_thread_num * sizeof (pthread_t));
  48. int i = 0;
  49. for (i = 0; i < max_thread_num; i++)
  50. {
  51. pthread_create (&(pool->threadid[i]), NULL, thread_routine,
  52. NULL);
  53. }
  54. }
  55. /*向线程池中加入任务*/
  56. int
  57. pool_add_worker (void *(*process) (void *arg), void *arg)
  58. {
  59. /*构造一个新任务*/
  60. CThread_worker *newworker =
  61. (CThread_worker *) malloc (sizeof (CThread_worker));
  62. newworker->process = process;
  63. newworker->arg = arg;
  64. newworker->next = NULL;/*别忘置空*/
  65. pthread_mutex_lock (&(pool->queue_lock));
  66. /*将任务加入到等待队列中*/
  67. CThread_worker *member = pool->queue_head;
  68. if (member != NULL)
  69. {
  70. while (member->next != NULL)
  71. member = member->next;
  72. member->next = newworker;
  73. }
  74. else
  75. {
  76. pool->queue_head = newworker;
  77. }
  78. assert (pool->queue_head != NULL);
  79. pool->cur_queue_size++;
  80. pthread_mutex_unlock (&(pool->queue_lock));
  81. /*好了,等待队列中有任务了,唤醒一个等待线程;
  82. 注意如果所有线程都在忙碌,这句没有任何作用*/
  83. pthread_cond_signal (&(pool->queue_ready));
  84. return 0;
  85. }
  86. /*销毁线程池,等待队列中的任务不会再被执行,但是正在运行的线程会一直
  87. 把任务运行完后再退出*/
  88. int
  89. pool_destroy ()
  90. {
  91. if (pool->shutdown)
  92. return -1;/*防止两次调用*/
  93. pool->shutdown = 1;
  94. /*唤醒所有等待线程,线程池要销毁了*/
  95. pthread_cond_broadcast (&(pool->queue_ready));
  96. /*阻塞等待线程退出,否则就成僵尸了*/
  97. int i;
  98. for (i = 0; i < pool->max_thread_num; i++)
  99. pthread_join (pool->threadid[i], NULL);
  100. free (pool->threadid);
  101. /*销毁等待队列*/
  102. CThread_worker *head = NULL;
  103. while (pool->queue_head != NULL)
  104. {
  105. head = pool->queue_head;
  106. pool->queue_head = pool->queue_head->next;
  107. free (head);
  108. }
  109. /*条件变量和互斥量也别忘了销毁*/
  110. pthread_mutex_destroy(&(pool->queue_lock));
  111. pthread_cond_destroy(&(pool->queue_ready));
  112. free (pool);
  113. /*销毁后指针置空是个好习惯*/
  114. pool=NULL;
  115. return 0;
  116. }
  117. void *
  118. thread_routine (void *arg)
  119. {
  120. printf ("starting thread 0x%x\n", pthread_self ());
  121. while (1)
  122. {
  123. pthread_mutex_lock (&(pool->queue_lock));
  124. /*如果等待队列为0并且不销毁线程池,则处于阻塞状态; 注意
  125. pthread_cond_wait是一个原子操作,等待前会解锁,唤醒后会加锁*/
  126. while (pool->cur_queue_size == 0 && !pool->shutdown)
  127. {
  128. printf ("thread 0x%x is waiting\n", pthread_self ());
  129. pthread_cond_wait (&(pool->queue_ready), &(pool->queue_lock));
  130. }
  131. /*线程池要销毁了*/
  132. if (pool->shutdown)
  133. {
  134. /*遇到break,continue,return等跳转语句,千万不要忘记先解锁*/
  135. pthread_mutex_unlock (&(pool->queue_lock));
  136. printf ("thread 0x%x will exit\n", pthread_self ());
  137. pthread_exit (NULL);
  138. }
  139. printf ("thread 0x%x is starting to work\n", pthread_self ());
  140. /*assert是调试的好帮手*/
  141. assert (pool->cur_queue_size != 0);
  142. assert (pool->queue_head != NULL);
  143. /*等待队列长度减去1,并取出链表中的头元素*/
  144. pool->cur_queue_size--;
  145. CThread_worker *worker = pool->queue_head;
  146. pool->queue_head = worker->next;
  147. pthread_mutex_unlock (&(pool->queue_lock));
  148. /*调用回调函数,执行任务*/
  149. (*(worker->process)) (worker->arg);
  150. free (worker);
  151. worker = NULL;
  152. }
  153. /*这一句应该是不可达的*/
  154. pthread_exit (NULL);
  155. }

下面是测试代码

  1. void *
  2. myprocess (void *arg)
  3. {
  4. printf ("threadid is 0x%x, working on task %d\n", pthread_self (),*(int *) arg);
  5. sleep (1);/*休息一秒,延长任务的执行时间*/
  6. return NULL;
  7. }
  8. int
  9. main (int argc, char **argv)
  10. {
  11. pool_init (3);/*线程池中最多三个活动线程*/
  12. /*连续向池中投入10个任务*/
  13. int *workingnum = (int *) malloc (sizeof (int) * 10);
  14. int i;
  15. for (i = 0; i < 10; i++)
  16. {
  17. workingnum[i] = i;
  18. pool_add_worker (myprocess, &workingnum[i]);
  19. }
  20. /*等待所有任务完成*/
  21. sleep (5);
  22. /*销毁线程池*/
  23. pool_destroy ();
  24. free (workingnum);
  25. return 0;
  26. }

将上述所有代码放入threadpool.c文件中,
在Linux输入编译命令
$ gcc -o threadpool threadpool.c -lpthread

以下是运行结果
starting thread 0xb7df6b90
thread 0xb7df6b90 is waiting
starting thread 0xb75f5b90
thread 0xb75f5b90 is waiting
starting thread 0xb6df4b90
thread 0xb6df4b90 is waiting
thread 0xb7df6b90 is starting to work
threadid is 0xb7df6b90, working on task 0
thread 0xb75f5b90 is starting to work
threadid is 0xb75f5b90, working on task 1
thread 0xb6df4b90 is starting to work
threadid is 0xb6df4b90, working on task 2
thread 0xb7df6b90 is starting to work
threadid is 0xb7df6b90, working on task 3
thread 0xb75f5b90 is starting to work
threadid is 0xb75f5b90, working on task 4
thread 0xb6df4b90 is starting to work
threadid is 0xb6df4b90, working on task 5
thread 0xb7df6b90 is starting to work
threadid is 0xb7df6b90, working on task 6
thread 0xb75f5b90 is starting to work
threadid is 0xb75f5b90, working on task 7
thread 0xb6df4b90 is starting to work
threadid is 0xb6df4b90, working on task 8
thread 0xb7df6b90 is starting to work
threadid is 0xb7df6b90, working on task 9
thread 0xb75f5b90 is waiting
thread 0xb6df4b90 is waiting
thread 0xb7df6b90 is waiting
thread 0xb75f5b90 will exit
thread 0xb6df4b90 will exit
thread 0xb7df6b90 will exit

Linux平台下线程池的原理及实现的更多相关文章

  1. linux平台下防火墙iptables原理(转)

    原文地址:http://www.cnblogs.com/ggjucheng/archive/2012/08/19/2646466.html iptables简介 netfilter/iptables( ...

  2. linux平台下防火墙iptables原理

    iptables简单介绍 netfilter/iptables(简称为iptables)组成Linux平台下的包过滤防火墙,与大多数的Linux软件一样,这个包过滤防火墙是免费的,它能够取代昂贵的商业 ...

  3. Thrift在Windows及Linux平台下的安装和使用示例

    本文章也同时发表在个人博客Thrift在Windows及Linux平台下的安装和使用示例上. thrift介绍 Apache Thrift 是 Facebook 实现的一种高效的.支持多种编程语言的R ...

  4. Java线程池的原理及几类线程池的介绍

    刚刚研究了一下线程池,如果有不足之处,请大家不吝赐教,大家共同学习.共同交流. 在什么情况下使用线程池? 单个任务处理的时间比较短 将需处理的任务的数量大 使用线程池的好处: 减少在创建和销毁线程上所 ...

  5. Python 线程池的原理和实现及subprocess模块

    最近由于项目需要一个与linux shell交互的多线程程序,需要用python实现,之前从没接触过python,这次匆匆忙忙的使用python,发现python确实语法非常简单,功能非常强大,因为自 ...

  6. 基于Linux/C++简单线程池的实现

    我们知道Java语言对于多线程的支持十分丰富,JDK本身提供了很多性能优良的库,包括ThreadPoolExecutor和ScheduleThreadPoolExecutor等.C++11中的STL也 ...

  7. Java线程池实现原理及其在美团业务中的实践

    本文转载自Java线程池实现原理及其在美团业务中的实践 导语 随着计算机行业的飞速发展,摩尔定律逐渐失效,多核CPU成为主流.使用多线程并行计算逐渐成为开发人员提升服务器性能的基本武器.J.U.C提供 ...

  8. 转:一个跨WINDOWS LINUX平台的线程类

     来源:http://blog.csdn.net/dengxu11/article/details/7232681 继Windows下实现一个CThread封装类之后,这里我再实现一个跨WINDOWS ...

  9. Java 线程池(ThreadPoolExecutor)原理分析与使用

    在我们的开发中"池"的概念并不罕见,有数据库连接池.线程池.对象池.常量池等等.下面我们主要针对线程池来一步一步揭开线程池的面纱. 使用线程池的好处 1.降低资源消耗 可以重复利用 ...

随机推荐

  1. SQL语句执行时间测试

    select getdate()--开始执行时间   要执行的SQL语句 select getdate() --结束时间

  2. 转:python webdriver API 之下拉框处理

    下拉框也是 web 页面上非常常见的功能,webdriver 对于一般的下拉框处理起来也相当简单,要想定位下拉框中的内容,首先需要定位到下拉框:这样的二次定位,我们在前面的例子中已经有过使用,下面通过 ...

  3. HttpContext.Current 的缺陷

    了解ASP.NET的开发人员都知道它有个非常强大的对象 HttpContext,而且为了方便,ASP.NET还为它提供了一个静态属性HttpContext.Current来访问它,今天的博客打算就从H ...

  4. adb devices 显示error

    1.adb kill-server 2.adb start-server

  5. PHP内核探索:哈希碰撞攻击是什么?

    最近哈希表碰撞攻击(Hashtable collisions as DOS attack)的话题不断被提起,各种语言纷纷中招.本文结合PHP内核源码,聊一聊这种攻击的原理及实现. 哈希表碰撞攻击的基本 ...

  6. . ToString(),Convert.ToString(),(string),as比较:

    http://www.cnblogs.com/chehaoj/archive/2010/02/23/1671955.html 通常 object 到 string 有四种方式(假设有object ob ...

  7. 前端构建工具gulp入门

    本文假设你之前没有用过任何任务脚本(task runner)和命令行工具,一步步教你上手Gulp.不要怕,它其实很简单,我会分为五步向你介绍gulp并帮助你完成一些惊人的事情.那就直接开始吧. 第一步 ...

  8. sql 去除列中内容的单引号

    update  [dbo].[历史数据补全$]  set  bankCardNo=replace(bankCardNo,'''','') 总共4个单引号,外边2个是字符串固有的2个,里边两个就表示是一 ...

  9. PAT乙级 1020. 月饼 (25)(只得到23分)

    1020. 月饼 (25) 时间限制 100 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 CHEN, Yue 月饼是中国人在中秋佳节时吃的一种传统食 ...

  10. 在IIS站点中Adomd.net集成认证账号问题

    最近在做一个Asp.net项目的时候 ,在C#代码里面用到了Adomd.net去连接SSAS服务器做MDX查询,开发完成后将Asp.net代码部署到IIS后发现Adomd.net老是连接不到SSAS服 ...