免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 4011 | 回复: 9
打印 上一主题 下一主题

用C写的一个线程池模型,和大家共享,欢迎指点 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2005-07-20 10:14 |只看该作者 |倒序浏览

  1. /*******************************************************
  2. * HdrFileName:  thrdpool.h
  3. * Description:  Implement a thread pool.
  4. * CreateDate :  2005.7.16
  5. * Author     :  peijianzhong
  6. * Email      :  pjz0311@emails.bjut.edu.cn
  7. * Version    :  1.0
  8. * Interface  :
  9. *   thrdpool_t *createThrdPool( int maxthrds, int maxclts, char *plogfile );
  10. *   int addClt( thrdpool_t *pthrdpool, int cltid, thrdcall_t thrdcall, thrdcall_t thrdexit );
  11. *   void destroyThrdPool( thrdpool_t *pthrdpool );
  12. *******************************************************/

  13. #ifndef _THRDPOOL_H
  14. #define _THRDPOOL_H

  15. #include <pthread.h>;
  16. #include <stdlib.h>;
  17. #include "runlog.h"

  18. #define THRDPOOL_OK     0
  19. #define ETOOMANYCLTS   -1

  20. typedef struct {
  21.         pthread_t *pthrds;
  22.         int count;
  23. } thrdtab_t;

  24. typedef enum { IDLE, WAITING, PROCESSING } cltstate_t;
  25. typedef void *(*thrdfunc_t)(void *);

  26. /*  */
  27. typedef struct {
  28.         thrdfunc_t thrdfunc;
  29.         void *arg;
  30. } thrdcall_t;

  31. typedef struct {
  32.         int cltid;
  33.         cltstate_t state;        /* the current state of this client */
  34.         thrdcall_t thrdcall;        /* the function excuted by thread defined by user */
  35.         thrdcall_t thrdexit;        /* the function excuted by thread when process finishes */
  36.         long pri;        /* priority */
  37. } clt_t;

  38. typedef struct {
  39.         clt_t *pclts;
  40.         int count;                        /* the max count of clients permited */
  41.         int waitcount;        /* the count of clients in WAITING state */
  42.         int proscount;        /* the count of clients in PROCESSING state */
  43.         long prino;                        /* increasing number, used to set the coming client's priority */
  44. } clttab_t;

  45. typedef struct {
  46.         thrdtab_t thrdtab;
  47.         clttab_t clttab;        /*  */
  48.         pthread_mutex_t thrdmutex;/*  */
  49.         pthread_cond_t thrdcond;        /*  */
  50.         char logfile[256];        /* run log file name */
  51. } thrdpool_t;

  52. extern thrdpool_t *createThrdPool( int maxthrds, int maxclts, char *plogfile );
  53. extern int addClt( thrdpool_t *pthrdpool, int cltid, thrdcall_t thrdcall, thrdcall_t thrdexit );
  54. extern void destroyThrdPool( thrdpool_t *pthrdpool );

  55. #endif

  56. ------------------------------------------------------------------------------
  57. /*******************************************************
  58. * SrcFileName:  thrdpool.c
  59. * Description:  implement a thread pool.
  60. * CreateDate :  2005.7.16
  61. * Author     :  peijianzhong
  62. * Email      :  pjz0311@emails.bjut.edu.cn
  63. * Version    :  1.0
  64. * Interface  :
  65. *   thrdpool_t *createThrdPool( int maxthrds, int maxclts, char *plogfile );
  66. *   int addClt( thrdpool_t *pthrdpool, int cltid, thrdcall_t thrdcall, thrdcall_t thrdexit );
  67. *   void destroyThrdPool( thrdpool_t *pthrdpool );
  68. *******************************************************/

  69. #include "thrdpool.h"

  70. /******************************************************
  71. * FuncName    : thrdproc
  72. * Description :
  73. *                        provide a framework to do the client side real actions
  74. *                        it's blocked when no client arive.
  75. *
  76. * Input       : void *arg, actually it's a pointer to struct thrdpool_t
  77. * Output      : no
  78. * Return      : viod
  79. *****************************************************/
  80. static void *thrdproc( void *arg ) {
  81.         thrdpool_t *pthrdpool = ( thrdpool_t *)arg;
  82.         clt_t *clts = pthrdpool->;clttab.pclts;
  83.         pthread_detach( pthread_self() );
  84.         pthread_setcanceltype( PTHREAD_CANCEL_ASYNCHRONOUS, NULL );
  85.         for(;;) {
  86.                 int retlock;
  87.                 retlock = pthread_mutex_lock( &pthrdpool->;thrdmutex );
  88.                 while( pthrdpool->;clttab.waitcount == 0 )
  89.                         pthread_cond_wait( &pthrdpool->;thrdcond, &pthrdpool->;thrdmutex );
  90.                 /* find one client which has the highest priority */
  91.                 int idx, cltidx = -1;
  92.                 long hpri = 99999;
  93.                 for( idx = 0; idx < pthrdpool->;clttab.count; idx++ ) {
  94.                         if( clts[idx].state == WAITING ) {
  95.                                 if( hpri >; clts[idx].pri ) {
  96.                                         hpri = clts[idx].pri;
  97.                                         cltidx = idx;
  98.                                 }
  99.                         }
  100.                 }
  101.                 runLog(pthrdpool->;logfile,"found wait client, clientidx=%d, hpri=%ld, cltid=%d, state=%d, pri=%ld, current thread is %ld \n", cltidx,hpri, clts[cltidx].cltid, clts[cltidx].state, clts[cltidx].pri, pthread_self());
  102.                 clts[cltidx].state = PROCESSING;
  103.                 pthrdpool->;clttab.waitcount--;
  104.                 pthrdpool->;clttab.proscount++;
  105.                 //runLog(pthrdpool->;logfile,"clientidx=%d, cltid=%d, state=%d, pri=%ld, waitcount=%d \n", cltidx, clts[cltidx].cltid, clts[cltidx].state, clts[cltidx].pri, pthrdpool->;clttab.waitcount);
  106.                 pthread_mutex_unlock( &pthrdpool->;thrdmutex );
  107.                
  108.                 runLog( pthrdpool->;logfile, "start processing client [%d], current thread is %ld.\n", clts[cltidx].cltid, pthread_self() );
  109.                 /* processing area */
  110.                 ( *clts[cltidx].thrdcall.thrdfunc)( clts[cltidx].thrdcall.arg );
  111.                 runLog( pthrdpool->;logfile, "processing client [%d] end, current thread is %ld.\n", clts[cltidx].cltid, pthread_self());
  112.                 if( clts[cltidx].thrdexit.thrdfunc != NULL )
  113.                         ( *clts[cltidx].thrdexit.thrdfunc)( clts[cltidx].thrdexit.arg );
  114.                 /* free this client resource */
  115.                 pthread_mutex_lock( &pthrdpool->;thrdmutex );
  116.                 clts[cltidx].state = IDLE;
  117.                 pthrdpool->;clttab.proscount--;
  118.                 clts[cltidx].thrdcall.thrdfunc = NULL;
  119.                 clts[cltidx].thrdcall.arg = NULL;
  120.                 clts[cltidx].thrdexit.thrdfunc = NULL;
  121.                 clts[cltidx].thrdcall.arg = NULL;
  122.                 runLog( pthrdpool->;logfile, " the resource released,cltidx=%d, cltid=%d , pri=%ld, curthread is %ld \n", cltidx, clts[cltidx].cltid, clts[cltidx].pri, pthread_self() );
  123.                 pthread_mutex_unlock( &pthrdpool->;thrdmutex );
  124.         }
  125.        
  126.         return;
  127. }

  128. /******************************************************
  129. * FuncName    : createThrdPool
  130. * Description : To create a thread pool
  131. * Input       :
  132. *                        @maxthrds  : max thread count permit to create
  133. *                        @maxclts   : max clients count permit to serve
  134. *                        @plogfile  : log file name
  135. * Output      : no
  136. * Return      : the pointer of thrdpool_t instance created here.
  137. *****************************************************/

  138. thrdpool_t *createThrdPool( int maxthrds, int maxclts, char *plogfile ) {
  139.         int idx;
  140.         thrdpool_t *pthrdpool;
  141.         clt_t *clts;
  142.         /*
  143.          * create a thread pool and initialize it
  144.          * according to the parameters specified by the caller
  145.          */
  146.         if(( maxthrds <= 0 ) || ( maxclts <= 0 )) {
  147.                 runLog( plogfile, "smaller than zero is illegal for the paramter maxthrds or maxclts of function createThrdPool.\n" );
  148.                 return NULL;
  149.         }
  150.         pthrdpool = calloc( sizeof( thrdpool_t ), 1);
  151.         pthrdpool->;thrdtab.pthrds = calloc( sizeof( pthread_t ), maxthrds );
  152.         pthrdpool->;thrdtab.count = maxthrds;
  153.         pthrdpool->;clttab.pclts = calloc( sizeof( clt_t ), maxclts );
  154.         clts = pthrdpool->;clttab.pclts;
  155.         pthrdpool->;clttab.count = maxclts;
  156.         if( plogfile == NULL )
  157.                 pthrdpool->;logfile[0] = '\0';
  158.         else
  159.                 strcpy( pthrdpool->;logfile, plogfile );
  160.         pthrdpool->;clttab.waitcount = 0;        /* waiting count */
  161.         pthrdpool->;clttab.proscount = 0;        /* processing count */
  162.         pthrdpool->;clttab.prino = 0;                        /* increasing pri number */
  163.         for( idx = 0; idx < maxclts; idx++ ) {
  164.                 clts[idx].state = IDLE;
  165.                 clts[idx].pri = 0;
  166.                 clts[idx].cltid = -1;
  167.                 clts[idx].thrdcall.thrdfunc = NULL;
  168.                 clts[idx].thrdcall.arg = NULL;
  169.                 clts[idx].thrdexit.thrdfunc = NULL;
  170.                 clts[idx].thrdcall.arg = NULL;
  171.         }
  172.         pthread_mutex_init( &pthrdpool->;thrdmutex, NULL );
  173.         pthread_cond_init( &pthrdpool->;thrdcond, NULL );
  174.        
  175.         for( idx = 0; idx < maxthrds; idx++ ) {
  176.                 pthread_create( &pthrdpool->;thrdtab.pthrds[idx], NULL, thrdproc, (void *)pthrdpool );
  177.         }
  178.         runLog( plogfile, "new threadpool created, max threads count is %d, max clients count is %d.\n", maxthrds, maxclts );
  179.         return pthrdpool;
  180. }

  181. /******************************************************
  182. * FuncName    : destroyThrdPool
  183. * Description : free the resources the thread pool used
  184. * Input       : thrdpool_t *pthrdpool
  185. * Output      : no
  186. * Return      : void
  187. *****************************************************/

  188. void destroyThrdPool( thrdpool_t *pthrdpool ) {
  189.         int idx;
  190.         for( idx = 0; idx < pthrdpool->;thrdtab.count; idx++) {
  191.                 pthread_cancel( pthrdpool->;thrdtab.pthrds[idx] );
  192.         }
  193.         pthread_mutex_unlock( &pthrdpool->;thrdmutex );        /* maybe return error EPERM here, but it doesn't matter. */
  194.         free( pthrdpool->;thrdtab.pthrds );
  195.         free( pthrdpool->;clttab.pclts );
  196.         free( pthrdpool );
  197.         pthrdpool = NULL;
  198.        
  199. }

  200. /******************************************************
  201. * FuncName    : addClt
  202. * Description : add a new client to the pool specified by the param pthrdpool
  203. * Input       :
  204. *                        @pthrdpool : the thread pool to add the client to
  205. *                        @cltid     : the identifier used by user to identify the client,
  206. *                                                                         for example,it can be the connection descriptor returned by accept(..) call.
  207. *                        @thrdcall  : a struct include the real client function and it's arguments to perform by the thread
  208. *                        @thrdexit  : a struct include the real client function and it's arguments to perform when thrdcall finished.
  209. * Output      : no
  210. * Return      :
  211. *                        ETOOMANYCLTS : no place for the new client
  212. *                        THRDPOOL_OK  : the client is successfully added into the thread pool
  213. *****************************************************/

  214. int addClt( thrdpool_t *pthrdpool, int cltid, thrdcall_t thrdcall, thrdcall_t thrdexit ) {
  215.         int idx, idleidx = -1;
  216.         clt_t *clts;
  217.         pthread_mutex_lock( &pthrdpool->;thrdmutex );
  218.         /* get first idle client item        */
  219.         for( idx = 0; idx < pthrdpool->;clttab.count; idx++ ) {
  220.                 if( pthrdpool->;clttab.pclts[idx].state == IDLE ) {
  221.                         idleidx = idx;
  222.                         break;
  223.                 }
  224.         }
  225.         if( idleidx == -1 ) {       
  226.                 runLog( pthrdpool->;logfile, "too many clients for new client [%d].\n", cltid );
  227.                 pthread_mutex_unlock( &pthrdpool->;thrdmutex );
  228.                 return ETOOMANYCLTS;
  229.         }
  230.         clts = pthrdpool->;clttab.pclts;
  231.         clts[idleidx].cltid = cltid;
  232.         clts[idleidx].state = WAITING;
  233.         clts[idleidx].thrdcall.thrdfunc = thrdcall.thrdfunc;
  234.         clts[idleidx].thrdcall.arg = thrdcall.arg;
  235.         clts[idleidx].thrdexit.thrdfunc = thrdexit.thrdfunc;
  236.         clts[idleidx].thrdexit.arg = thrdexit.arg;
  237.         pthrdpool->;clttab.waitcount++;
  238.         /* calculate the priority of the coming client */
  239.         pthrdpool->;clttab.prino++;
  240.         clts[idleidx].pri = pthrdpool->;clttab.prino;
  241.         runLog( pthrdpool->;logfile, "new client [%d] added,cltidx=%d, state=%d, pri=%ld now waiting for service.\n", cltid,idleidx,clts[idleidx].state,clts[idleidx].pri );
  242.         if( pthrdpool->;clttab.waitcount == 1 )
  243.                 pthread_cond_signal( &pthrdpool->;thrdcond );
  244.         pthread_mutex_unlock( &pthrdpool->;thrdmutex );
  245.         return THRDPOOL_OK;
  246. }
  247. ----------------------------------------------------------
  248. int main() {
  249.         int listenfd, connfd, saidx = 0, *pconnfd;
  250.         rspns_t rspns;
  251.         char msg[256];
  252.         thrdpool_t *pthrdpool;
  253.         thrdcall_t thrdcall, thrdexit = { NULL, NULL };
  254.         initNsTab();
  255.         if(( pthrdpool = createThrdPool( MAXTHREADS, MAXCLIENTS, NS_RUNLOGFILE )) == NULL )
  256.                 return -1;
  257.        
  258.         bzero( &rspns, sizeof( rspns_t ) );
  259.         if(( saidx = findSaRcd( NAMESERVID )) == ERR ) {
  260.                 sprintf( msg, "Naming Server Conf Info [%s] not found in [%s].\n", NAMESERVID, SA_DOMAINCONF );
  261.                 errLog( NS_RUNLOGFILE, msg );
  262.                 exit(ERR);
  263.         }
  264.         listenfd = tcpListen( atoi( satab.psa[saidx].sa_port ), LISTENQ );
  265.         runLog( NS_RUNLOGFILE, "Naming Server is listening in port [%s] now. It's listenfd is %d.\n", satab.psa[saidx].sa_port, listenfd );
  266.         for(;;) {
  267.                 if(( connfd = tcpAccept( listenfd ) ) < 0 ) {
  268.                         errLog( NS_RUNLOGFILE, "accept error.\n" );
  269.                         exit(1);
  270.                 }
  271.                 pconnfd = (int *)malloc( sizeof( int ) );
  272.                 *pconnfd = connfd;
  273.                 thrdcall.thrdfunc = doResponse;
  274.                 thrdcall.arg = pconnfd;
  275.                 addClt( pthrdpool, connfd, thrdcall, thrdexit );
  276.         }
  277.        
  278.         return OK;
  279. }
复制代码
[/code]

论坛徽章:
1
荣誉版主
日期:2011-11-23 16:44:17
2 [报告]
发表于 2005-07-20 10:15 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

加上CODE标签好吗~?
谢谢LZ .

论坛徽章:
0
3 [报告]
发表于 2005-07-20 10:18 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

sorry, 没想到会这样,告诉我该怎么加code标签好吗

论坛徽章:
1
荣誉版主
日期:2011-11-23 16:44:17
4 [报告]
发表于 2005-07-20 10:20 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

你重新编辑一下 你的帖子 在代码的前面加上[ CODE ] 后面加上 [ / CODE ].. 加上缩进 大家能看的清楚些.

论坛徽章:
0
5 [报告]
发表于 2005-07-20 11:16 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

谢谢楼主共享
先收下了
回家慢慢看~
^_^

论坛徽章:
0
6 [报告]
发表于 2005-07-23 20:49 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

线程池要解决的问题:避免频繁的创建和销毁线程,类似于内存池,连接池等等概念
线程池的关键思想:线程可重复利用
实现方法:当一个线程处理完一件事之后,将它挂起直到有新的任务再将它激活

明白了这一点实现起来就简单明了了

论坛徽章:
0
7 [报告]
发表于 2005-07-24 11:25 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

3Q

论坛徽章:
0
8 [报告]
发表于 2005-07-24 12:59 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

好长。。跟《unp(第2版)》27.11节的预先建立线程池的服务端有何区别?

论坛徽章:
0
9 [报告]
发表于 2005-07-24 22:13 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

注意:
楼主的代码是不错,不过在unix 下 无法编译

论坛徽章:
0
10 [报告]
发表于 2005-07-25 10:37 |只看该作者

用C写的一个线程池模型,和大家共享,欢迎指点

THREAD POOL也没那么简单,一般比较流行的是LEADER/FLOWER模型。效率要高一些。。详细的可以看ACE的发起人写的那本书。。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP