一个简单的linux线程池

http://blog.csdn.net/wzjking0929/article/details/20312675

线程池:简单地说,线程池 就是预先创建好一批线程,方便、快速地处理收到的业务。比起传统的到来一个任务,即时创建一个线程来处理,节省了线程的创建和回收的开销,响应更快,效率更高。

 

在linux中,使用的是posix线程库,首先介绍几个常用的函数:

1 线程的创建和取消函数

pthread_create

创建线程

pthread_join

合并线程

pthread_cancel

取消线程

2 线程同步函数

pthread_mutex_lock

pthread_mutex_unlock

pthread_cond_signal

pthread_cond_wait

 

关于函数的详细说明,参考man手册

 

线程池的实现:

线程池的实现主要分为三部分,线程的创建、添加任务到线程池中、工作线程从任务队列中取出任务进行处理。

主要有两个类来实现,CTask,CThreadPool

/**
执行任务的类,设置任务数据并执行
**/

C代码  收藏代码
  1. class CTask  
  2. {  
  3. protected:  
  4.  string m_strTaskName;  //任务的名称  
  5.  void* m_ptrData;       //要执行的任务的具体数据  
  6. public:  
  7.  CTask(){}  
  8.  CTask(string taskName)  
  9.  {  
  10.   this->m_strTaskName = taskName;  
  11.   m_ptrData = NULL;  
  12.  }  
  13.  virtual int Run()= 0;  
  14.  void SetData(void* data);    //设置任务数据  
  15. };  

 

任务类是个虚类,所有的任务要从CTask类中继承 ,实现run接口,run接口中需要实现的就是具体解析任务的逻辑。m_ptrData是指向任务数据的指针,可以是简单数据类型,也可以是自定义的复杂数据类型。

 

线程池类

/**
线程池
**/

Java代码  收藏代码
  1. class CThreadPool  
  2. {  
  3. private:  
  4.  vector<CTask*> m_vecTaskList;         //任务列表  
  5.  int m_iThreadNum;                            //线程池中启动的线程数             
  6.  static vector<pthread_t> m_vecIdleThread;   //当前空闲的线程集合  
  7.  static vector<pthread_t> m_vecBusyThread;   //当前正在执行的线程集合  
  8.  static pthread_mutex_t m_pthreadMutex;    //线程同步锁  
  9.  static pthread_cond_t m_pthreadCond;    //线程同步的条件变量  
  10. protected:  
  11.  static void* ThreadFunc(void * threadData); //新线程的线程函数  
  12.  static int MoveToIdle(pthread_t tid);   //线程执行结束后,把自己放入到空闲线程中  
  13.  static int MoveToBusy(pthread_t tid);   //移入到忙碌线程中去  
  14.  int Create();          //创建所有的线程  
  15. public:  
  16.  CThreadPool(int threadNum);  
  17.  int AddTask(CTask *task);      //把任务添加到线程池中  
  18.  int StopAll();  
  19. };  

 

当线程池对象创建后,启动一批线程,并把所有的线程放入空闲列表中,当有任务到达时,某一个线程取出任务并进行处理。

线程之间的同步用线程锁和条件变量。

这个类的对外接口有两个:

AddTask函数把任务添加到线程池的任务列表中,并通知线程进行处理。当任务到到时,把任务放入m_vecTaskList任务列表中,并用pthread_cond_signal唤醒一个线程进行处理。

StopAll函数停止所有的线程

 

Cpp代码  收藏代码
  1. ************************************************  
  2.   
  3. 代码:  
  4.   
  5. ××××××××××××××××××××CThread.h  
  6.   
  7.    
  8.   
  9. #ifndef __CTHREAD  
  10. #define __CTHREAD  
  11. #include <vector>  
  12. #include <string>  
  13. #include <pthread.h>  
  14.   
  15. using namespace std;  
  16.   
  17. /** 
  18. 执行任务的类,设置任务数据并执行 
  19. **/  
  20. class CTask  
  21. {  
  22. protected:  
  23.  string m_strTaskName;  //任务的名称  
  24.  void* m_ptrData;       //要执行的任务的具体数据  
  25. public:  
  26.  CTask(){}  
  27.  CTask(string taskName)  
  28.  {  
  29.   this->m_strTaskName = taskName;  
  30.   m_ptrData = NULL;  
  31.  }  
  32.  virtual int Run()= 0;  
  33.  void SetData(void* data);    //设置任务数据  
  34. };  
  35.   
  36. /** 
  37. 线程池 
  38. **/  
  39. class CThreadPool  
  40. {  
  41. private:  
  42.  vector<CTask*> m_vecTaskList;         //任务列表  
  43.  int m_iThreadNum;                            //线程池中启动的线程数             
  44.  static vector<pthread_t> m_vecIdleThread;   //当前空闲的线程集合  
  45.  static vector<pthread_t> m_vecBusyThread;   //当前正在执行的线程集合  
  46.  static pthread_mutex_t m_pthreadMutex;    //线程同步锁  
  47.  static pthread_cond_t m_pthreadCond;    //线程同步的条件变量  
  48. protected:  
  49.  static void* ThreadFunc(void * threadData); //新线程的线程函数  
  50.  static int MoveToIdle(pthread_t tid);   //线程执行结束后,把自己放入到空闲线程中  
  51.  static int MoveToBusy(pthread_t tid);   //移入到忙碌线程中去  
  52.  int Create();          //创建所有的线程  
  53. public:  
  54.  CThreadPool(int threadNum);  
  55.  int AddTask(CTask *task);      //把任务添加到线程池中  
  56.  int StopAll();  
  57. };  
  58.   
  59. #endif  
  60.   
  61.    
  62.   
  63.    
  64.   
  65.    
  66.   
  67. 类的实现为:  
  68.   
  69. ××××××××××××××××××××CThread.cpp  
  70.   
  71.    
  72.   
  73. #include "CThread.h"  
  74. #include <string>  
  75. #include <iostream>  
  76.   
  77. using namespace std;  
  78.   
  79. void CTask::SetData(void * data)  
  80. {  
  81.  m_ptrData = data;  
  82. }  
  83.   
  84. vector<pthread_t> CThreadPool::m_vecBusyThread;  
  85. vector<pthread_t> CThreadPool::m_vecIdleThread;  
  86. pthread_mutex_t CThreadPool::m_pthreadMutex = PTHREAD_MUTEX_INITIALIZER;  
  87. pthread_cond_t CThreadPool::m_pthreadCond = PTHREAD_COND_INITIALIZER;  
  88.   
  89. CThreadPool::CThreadPool(int threadNum)  
  90. {  
  91.  this->m_iThreadNum = threadNum;  
  92.  Create();  
  93. }  
  94. int CThreadPool::MoveToIdle(pthread_t tid)  
  95. {  
  96.  vector<pthread_t>::iterator busyIter = m_vecBusyThread.begin();  
  97.  while(busyIter != m_vecBusyThread.end())  
  98.  {  
  99.   if(tid == *busyIter)  
  100.   {  
  101.    break;  
  102.   }  
  103.   busyIter++;  
  104.  }  
  105.  m_vecBusyThread.erase(busyIter);  
  106.  m_vecIdleThread.push_back(tid);  
  107.  return 0;  
  108. }  
  109.   
  110. int CThreadPool::MoveToBusy(pthread_t tid)  
  111. {  
  112.  vector<pthread_t>::iterator idleIter = m_vecIdleThread.begin();  
  113.  while(idleIter != m_vecIdleThread.end())  
  114.  {  
  115.   if(tid == *idleIter)  
  116.   {  
  117.    break;  
  118.   }  
  119.   idleIter++;  
  120.  }  
  121.  m_vecIdleThread.erase(idleIter);  
  122.  m_vecBusyThread.push_back(tid);  
  123.  return 0;  
  124. }  
  125. void* CThreadPool::ThreadFunc(void * threadData)  
  126. {  
  127.  pthread_t tid = pthread_self();  
  128.  while(1)  
  129.  {  
  130.   pthread_mutex_lock(&m_pthreadMutex);  
  131.   pthread_cond_wait(&m_pthreadCond,&m_pthreadMutex);  
  132.   cout << "tid:" << tid << " run" << endl;  
  133.   //get task  
  134.   vector<CTask*>* taskList = (vector<CTask*>*)threadData;  
  135.   vector<CTask*>::iterator iter = taskList->begin();  
  136.   while(iter != taskList->end())  
  137.   {  
  138.      
  139.    MoveToBusy(tid);  
  140.    break;  
  141.   }  
  142.   CTask* task = *iter;  
  143.   taskList->erase(iter);  
  144.   pthread_mutex_unlock(&m_pthreadMutex);  
  145.   cout << "idel thread number:" << CThreadPool::m_vecIdleThread.size() << endl;  
  146.   cout << "busy thread number:" << CThreadPool::m_vecBusyThread.size() << endl;  
  147.   //cout << "task to be run:" << taskList->size() << endl;  
  148.   task->Run();  
  149.     
  150.   //cout << "CThread::thread work" << endl;  
  151.   cout << "tid:" << tid << " idle" << endl;  
  152.     
  153.  }  
  154.  return (void*)0;  
  155. }  
  156.   
  157. int CThreadPool::AddTask(CTask *task)  
  158. {  
  159.  this->m_vecTaskList.push_back(task);  
  160.  pthread_cond_signal(&m_pthreadCond);  
  161.  return 0;  
  162. }  
  163. int CThreadPool::Create()  
  164. {  
  165.  for(int i = 0; i < m_iThreadNum;i++)  
  166.  {  
  167.   pthread_t tid = 0;  
  168.   pthread_create(&tid,NULL,ThreadFunc,&m_vecTaskList);  
  169.   m_vecIdleThread.push_back(tid);  
  170.  }  
  171.  return 0;  
  172. }  
  173.   
  174. int CThreadPool::StopAll()  
  175. {  
  176.  vector<pthread_t>::iterator iter = m_vecIdleThread.begin();  
  177.  while(iter != m_vecIdleThread.end())  
  178.  {  
  179.   pthread_cancel(*iter);  
  180.   pthread_join(*iter,NULL);  
  181.   iter++;  
  182.  }  
  183.   
  184.  iter = m_vecBusyThread.begin();  
  185.  while(iter != m_vecBusyThread.end())  
  186.  {  
  187.   pthread_cancel(*iter);  
  188.   pthread_join(*iter,NULL);  
  189.   iter++;  
  190.  }  
  191.    
  192.  return 0;  
  193. }  
  194.   
  195. 简单示例:  
  196.   
  197. ××××××××test.cpp  
  198.   
  199. #include "CThread.h"  
  200. #include <iostream>  
  201.   
  202. using namespace std;  
  203.   
  204. class CWorkTask: public CTask  
  205. {  
  206. public:  
  207.  CWorkTask()  
  208.  {}  
  209.  int Run()  
  210.  {  
  211.   cout << (char*)this->m_ptrData << endl;  
  212.   sleep(10);  
  213.   return 0;  
  214.  }  
  215. };  
  216. int main()  
  217. {  
  218.  CWorkTask taskObj;  
  219.  char szTmp[] = "this is the first thread running,haha success";  
  220.  taskObj.SetData((void*)szTmp);  
  221.  CThreadPool threadPool(10);  
  222.  for(int i = 0;i < 11;i++)  
  223.  {  
  224.   threadPool.AddTask(&taskObj);  
  225.  }  
  226.  while(1)  
  227.  {  
  228.   sleep(120);  
  229.  }  
  230.  return 0;  
  231. }  

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/383823.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【C++学习笔记三】C++多态、抽象(接口)

当类之间存在多种层次结构&#xff0c;并且类之间通过继承关联时就会用到多态。 虚函数在子类中的覆盖版本和该函数在基类中的原始版本必须有相同的函数签名、函数名、形参名、常属性。如果返回值为非类类型&#xff0c;则必须相同&#xff0c;如果是类类型A的指针或者引用&am…

C++重载和重写的条件以及重写后对基类函数的覆盖

重载&#xff1a;同一个类中名字相同&#xff0c;参数列表不同的方法构成重载函数&#xff0c;和返回值没有关系。这就意味着就算返回值不同&#xff0c;只要名字相同参数列表相同编译器还是会报错&#xff0c;觉得一函数被定义了两次。 重写&#xff1a;派生类中只要函数名字…

C++静态成员和静态方法

在类中&#xff0c;静态成员可以实现多个对象之间共享数据&#xff0c;同时保证了安全性。静态数据对该类的所有对象是公有的&#xff0c;存储一处供所有对象使用。 注意&#xff1a; 静态成员定义时需要在前面加上关键字static静态成员必须初始化且必须在类外进行&#xff0…

基于epoll的简单的http服务器

http://blog.csdn.net/fangjian1204/article/details/34415651 http服务器已经可以处理并发连接&#xff0c;支持多个客户端并发访问&#xff0c;每个连接可以持续读写数据&#xff0c;当然&#xff0c;这只是一个简单的学习例子&#xff0c;还有很多bug&#xff0c;发表出来只…

C++单例模式简单实现

有时候我们需要某个类只能被实例化一次&#xff0c;并且其他类都可以访问到这个类&#xff0c;就需要这种设计模式。 例如我们想要做个资源管理器&#xff0c;显然这个管理器只能有一个。 这种模式有很多实现方式&#xff0c;这里介绍最简单的一种&#xff0c;想要了解更多可…

Linux C++ 实现线程池

http://blog.csdn.net/qq_25425023/article/details/53914609 线程池中的线程&#xff0c;在任务队列为空的时候&#xff0c;等待任务的到来&#xff0c;任务队列中有任务时&#xff0c;则依次获取任务来执行&#xff0c;任务队列需要同步。 Linux线程同步有多种方法&#xff…

C++制表符

制表符的转义字符为\t&#xff0c;一般情况下长度为8个空格&#xff0c;这里的8个指的是从上一个字符串的开头开始算&#xff0c;往后数8个&#xff0c;不够的话就补空格。 如果前面的字符串的长度大于等于8个&#xff0c;例如前面字符串的长度为x,那么就会补(8-x%8)个空格 例…

C++派生类含有成员对象构造函数析构函数顺序

参考博客&#xff1a;传送门1 当类中含有对象成员时&#xff1a; 类的构造函数要包含对成员对象的初始化&#xff0c;如果构造函数的成员初始化列表没有包含对成员对象的初始化&#xff0c;系统会自动调用成员对象的无参构造函数。顺序上&#xff1a;先调用成员对象的构造函数…

c,c++中字符串处理函数strtok,strstr,strchr,strsub

http://blog.csdn.net/wangqing_12345/article/details/51760220 1&#xff0c;字符串切割函数 函数原型&#xff1a;char *strtok(char *s, char *delim); 函数功能&#xff1a;把字符串s按照字符串delim进行分割&#xff0c;然后返回分割的结果。 函数使用说&#xff1a; 1…

C++虚基类成员可见性

详见《CPrimer》[第五版]719页 如果继承路径上没有和虚基类成员重名的成员&#xff0c;则不存在二义性&#xff0c;因为我们仅能访问到虚基类成员。 当访问仅有一条继承路径上含有和虚基类成员重名的成员&#xff0c;也不存在二义性。派生类的成员的优先级比基类的成员高&…

链表逆序的原理及实例

http://blog.csdn.net/wangqing_12345/article/details/51757294 尾插法建立链表&#xff0c;带头结点设链表节点为typedef struct node {int data;struct node *next;}node_t, *pnode_t;要求将一带链表头List head的单向链表逆序。 分析&#xff1a; 1). 若链表为空或只有一个…

C++关于虚基类、构造函数、析构函数、成员对象的两个程序浅析

预备博客&#xff1a; C虚继承中构造函数和析构函数顺序问题以及原理 C派生类含有成员对象构造函数析构函数顺序 C虚基类成员可见性 程序一如下&#xff1a; #include<iostream> using namespace std; class A { public:A(int a) :x(a) { cout << "A const…

strtok函数及其实现

头文件&#xff1a;#include <string.h> 定义函数&#xff1a;char * strtok(char *s, const char *delim); 函数说明&#xff1a;strtok()用来将字符串分割成一个个片段。参数s 指向欲分割的字符串&#xff0c;参数delim 则为分割字符串&#xff0c;当 strtok()在参数s …

C++小型公司管理系统

项目要求&#xff1a; 编写一个程序实现小型公司的人员信息管理系统。该公司雇员&#xff08;employee&#xff09;包括经理&#xff08;manager&#xff09;&#xff0c;技术人员&#xff08;technician&#xff09;、销售员&#xff08;salesman&#xff09;和销售部经理&…

Linux网络编程“惊群”问题总结

http://www.cnblogs.com/Anker/p/7071849.html 1、前言 我从事Linux系统下网络开发将近4年了&#xff0c;经常还是遇到一些问题&#xff0c;只是知其然而不知其所以然&#xff0c;有时候和其他人交流&#xff0c;搞得非常尴尬。如今计算机都是多核了&#xff0c;网络编程框架也…

【Java学习笔记六】常用数据对象之String

字符串 在Java中系统定义了两种类型的字符串类&#xff1a;String和StringBuffer String类对象的值和长度都不能改变&#xff0c;称为常量字符串类&#xff0c;其中每个值称为常量字符串。 StringBuffer类对象的值和长度都可以改变&#xff0c;称为变量字符串类&#xff0c;其…

【Java学习笔记七】常用数据对象之数组

同一般的对象创建和定义一样&#xff0c;数组的定义和创建可以分开进行也可以合并一起进行。 一维数组定义格式&#xff1a; <元素类型>[] <数组名>;//[]也可以放在数组名的后面一维数组创建格式&#xff1a; new <元素类型>[<元素个数>];执行new运…

yfan.qiu linux硬链接与软链接

http://www.cnblogs.com/yfanqiu/archive/2012/06/11/2545556.html Linux 系统中有软链接和硬链接两种特殊的“文件”。 软链接可以看作是Windows中的快捷方式&#xff0c;可以让你快速链接到目标档案或目录。 硬链接则透过文件系统的inode来产生新档名&#xff0c;而不是产生…

【Java学习笔记八】包装类和vector

包装类 在Java语言中&#xff0c;每一种基本的数据类型都有相应的对象类型&#xff0c;称为他们基本类型的包装类&#xff08;包裹类&#xff09;。 字节byte&#xff1a;Byte、短整数型short&#xff1a;Short 标准整数型int&#xff1a;Integer、长整数型long&#xff1a;Lo…

Linux C++线程池实例

http://www.cnblogs.com/danxi/p/6636095.html 想做一个多线程服务器测试程序&#xff0c;因此参考了github的一些实例&#xff0c;然后自己动手写了类似的代码来加深理解。 目前了解的线程池实现有2种思路&#xff1a; 第一种&#xff1a; 主进程创建一定数量的线程&#xff0…