百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 优雅编程 > 正文

【Linux系统编程】线程池

sinye56 2025-01-03 18:45 9 浏览 0 评论

01. 线程池原理

在传统服务器结构中,常是有一个总的监听线程监听有没有新的用户连接服务器,每当有一个新的用户进入,服务器就开启一个新的线程用户处理这 个用户的数据包。这个线程只服务于这个用户,当用户与服务器端关闭连接以后,服务器端销毁这个线程。

然而频繁地开辟与销毁线程极大地占用了系统的资源,而且在大量用户的情况下,系统为了开辟和销毁线程将浪费大量的时间和资源。线程池提供了一个解决外部大量用户与服务器有限资源的矛盾。

线程池和传统的一个用户对应一个线程的处理方法不同,它的基本思想就是在程序开始时就在内存中开辟一些线程,线程的数目是固定的,他们独自形成一个类,屏蔽了对外的操作,而服务器只需要将数据包交给线程池就可以了。当有新的客户请求到达时,不是新创建一个线程为其服务,而是从“池子”中选择一个空闲的线程为新的客户请求服务,服务完毕后,线程进入空闲线程池中。如果没有线程空闲的话,就将数据包暂时积累, 等待线程池内有线程空闲以后再进行处理。通过对多个任务重用已经存在的线程对象,降低了对线程对象创建和销毁的开销。当客户请求 时,线程对象已经存在,可以提高请求的响应时间,从而整体地提高了系统服务的表现。

02. 线程池应用实例

一般来说实现一个线程池主要包括以下几个组成部分:

1)线程管理器:用于创建并管理线程池。

2)工作线程:线程池中实际执行任务的线程。在初始化线程时会预先创建好固定数目的线程在池中,这些初始化的线程一般处于空闲状态,一般不占用 CPU,占用较小的内存空间。

3)任务接口:每个任务必须实现的接口,当线程池的任务队列中有可执行任务时,被空闲的工作线程调去执行(线程的闲与忙是通过互斥量实现的),把任务抽象出来形成接口,可以做到线程池与具体的任务无关。

4)任务队列:用来存放没有处理的任务,提供一种缓冲机制,实现这种结构有好几种方法,常用的是队列,主要运用先进先出原理,另外一种是链表之类的数据结构,可以动态的为它分配内存空间,应用中比较灵活,此教程就是用到的链表。

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

03. 线程池代码

thread_pool.h的示例代码

#ifndef __THREAD_POOL_H__

#define __THREAD_POOL_H__


#include <pthread.h>


/*********************************************************************

* 任务回调函数,也可根据需要自行修改

*********************************************************************/

typedef void *(*pool_task_f)(void *arg);


/*********************************************************************

* 任务句柄

*********************************************************************/

typedef struct _task{

pool_task_f process;/*回调函数,任务运行时会调用此函数,注意也可声明成其它形式*/

void *arg; /*回调函数的参数*/

struct _task *next;

}pool_task;


/*********************************************************************

* 线程池句柄

*********************************************************************/

typedef struct

{

pthread_t *threadid; /* 线程号 */

int threads_limit; /* 线程池中允许的活动线程数目 */

int destroy_flag; /* 是否销毁线程池 , 0销毁,1不销毁*/

pool_task *queue_head; /* 链表结构,线程池中所有等待任务 */

int task_in_queue; /* 当前等待队列的任务数目 */

pthread_mutex_t queue_lock; /* 锁 */

pthread_cond_t queue_ready; /* 条件变量 */

}pool_t;


/*********************************************************************

*功能: 初始化线程池结构体并创建线程

*参数:

pool:线程池句柄

threads_limit:线程池中线程的数量

*返回值: 无

*********************************************************************/

void pool_init(pool_t *pool, int threads_limit);


/*********************************************************************

*功能: 销毁线程池,等待队列中的任务不会再被执行,

但是正在运行的线程会一直,把任务运行完后再退出

*参数: 线程池句柄

*返回值: 成功:0,失败非0

*********************************************************************/

int pool_uninit(pool_t *pool);


/*********************************************************************

*功能: 向线程池中添加一个任务

*参数:

pool:线程池句柄

process:任务处理函数

arg:任务参数

*返回值: 0

*********************************************************************/

int pool_add_task(pool_t *pool, pool_task_f process, void *arg);

#endif

thread_pool.c 的示例代码

#include <stdio.h>

#include <stdlib.h>

#include <pthread.h>

#include <assert.h>


#include "thread_pool.h"


static void *pool_thread_server(void *arg);


/*********************************************************************

*功能: 初始化线程池结构体并创建线程

*参数:

pool:线程池句柄

threads_limit:线程池中线程的数量

*返回值: 无

*********************************************************************/

void pool_init(pool_t *pool, int threads_limit)

{

pool->threads_limit = threads_limit;

pool->queue_head = NULL;

pool->task_in_queue = 0;

pool->destroy_flag = 0;

/*创建存放线程ID的空间*/

pool->threadid = (pthread_t *)calloc(threads_limit, sizeof(pthread_t));

int i = 0;

/*初始化互斥锁和条件变量*/

pthread_mutex_init(&(pool->queue_lock), NULL);

pthread_cond_init(&(pool->queue_ready), NULL);

/*循环创建threads_limit个线程*/

for (i = 0; i < threads_limit; i++){

pthread_create(&(pool->threadid[i]), NULL, pool_thread_server, pool);

}

return;

}


/*********************************************************************

*功能: 销毁线程池,等待队列中的任务不会再被执行,

但是正在运行的线程会一直,把任务运行完后再退出

*参数: 线程池句柄

*返回值: 成功:0,失败非0

*********************************************************************/

int pool_uninit(pool_t *pool)

{

pool_task *head = NULL;

int i;


pthread_mutex_lock(&(pool->queue_lock));

if(pool->destroy_flag)/* 防止两次调用 */

return -1;

pool->destroy_flag = 1;

pthread_mutex_unlock(&(pool->queue_lock));

/* 唤醒所有等待线程,线程池要销毁了 */

pthread_cond_broadcast(&(pool->queue_ready));

/* 阻塞等待线程退出,否则就成僵尸了 */

for (i = 0; i < pool->threads_limit; i++)

pthread_join(pool->threadid[i], NULL);

free(pool->threadid);

/* 销毁等待队列 */

pthread_mutex_lock(&(pool->queue_lock));

while(pool->queue_head != NULL){

head = pool->queue_head;

pool->queue_head = pool->queue_head->next;

free(head);

}

pthread_mutex_unlock(&(pool->queue_lock));

/*条件变量和互斥量也别忘了销毁*/

pthread_mutex_destroy(&(pool->queue_lock));

pthread_cond_destroy(&(pool->queue_ready));

return 0;

}


/*********************************************************************

*功能: 向任务队列中添加一个任务

*参数:

pool:线程池句柄

process:任务处理函数

arg:任务参数

*返回值: 无

*********************************************************************/

static void enqueue_task(pool_t *pool, pool_task_f process, void *arg)

{

pool_task *task = NULL;

pool_task *member = NULL;


pthread_mutex_lock(&(pool->queue_lock));


if(pool->task_in_queue >= pool->threads_limit){

printf("task_in_queue > threads_limit!\n");

pthread_mutex_unlock (&(pool->queue_lock));

return;

}


task = (pool_task *)calloc(1, sizeof(pool_task));

assert(task != NULL);

task->process = process;

task->arg = arg;

task->next = NULL;

pool->task_in_queue++;

member = pool->queue_head;

if(member != NULL){

while(member->next != NULL) /* 将任务加入到任务链连的最后位置. */

member = member->next;

member->next = task;

}else{

pool->queue_head = task; /* 如果是第一个任务的话,就指向头 */

}

printf("\ttasks %d\n", pool->task_in_queue);

/* 等待队列中有任务了,唤醒一个等待线程 */

pthread_cond_signal (&(pool->queue_ready));

pthread_mutex_unlock (&(pool->queue_lock));

}


/*********************************************************************

*功能: 从任务队列中取出一个任务

*参数: 线程池句柄

*返回值: 任务句柄

*********************************************************************/

static pool_task *dequeue_task(pool_t *pool)

{

pool_task *task = NULL;


pthread_mutex_lock(&(pool->queue_lock));

/* 判断线程池是否要销毁了 */

if(pool->destroy_flag){

pthread_mutex_unlock(&(pool->queue_lock));

printf("thread 0x%lx will be destroyed\n", pthread_self());

pthread_exit(NULL);

}

/* 如果等待队列为0并且不销毁线程池,则处于阻塞状态 */

if(pool->task_in_queue == 0){

while((pool->task_in_queue == 0) && (!pool->destroy_flag)){

printf("thread 0x%lx is waitting\n", pthread_self());

/* 注意:pthread_cond_wait是一个原子操作,等待前会解锁,唤醒后会加锁 */

pthread_cond_wait(&(pool->queue_ready), &(pool->queue_lock));

}

}else{

/* 等待队列长度减去1,并取出队列中的第一个元素 */

pool->task_in_queue--;

task = pool->queue_head;

pool->queue_head = task->next;

printf("thread 0x%lx received a task\n", pthread_self());

}

pthread_mutex_unlock(&(pool->queue_lock));

return task;

}


/*********************************************************************

*功能: 向线程池中添加一个任务

*参数:

pool:线程池句柄

process:任务处理函数

arg:任务参数

*返回值: 0

*********************************************************************/

int pool_add_task(pool_t *pool, pool_task_f process, void *arg)

{

enqueue_task(pool, process, arg);

return 0;

}


/*********************************************************************

*功能: 线程池服务程序

*参数: 略

*返回值: 略

*********************************************************************/

static void *pool_thread_server(void *arg)

{

pool_t *pool = NULL;


pool = (pool_t *)arg;

while(1){

pool_task *task = NULL;

task = dequeue_task(pool);

/*调用回调函数,执行任务*/

if(task != NULL){

printf ("thread 0x%lx is busy\n", pthread_self());

task->process(task->arg);

free(task);

task = NULL;

}

}

/*这一句应该是不可达的*/

pthread_exit(NULL);

return NULL;

}

测试代码:

#include <stdio.h>

#include <unistd.h>


#include "thread_pool.h"


void *task_test(void *arg)

{

printf("\t\tworking on task %d\n", (int)arg);

sleep(1); /*休息一秒,延长任务的执行时间*/

return NULL;

}


void thread_pool_demo(void)

{

pool_t pool;

int i = 0;


pool_init(&pool, 2);//初始化一个线程池,其中创建2个线程

sleep(1);

for(i = 0; i < 5; i++){

sleep(1);

pool_add_task(&pool, task_test, (void *)i);//添加一个任务

}

sleep(4);


pool_uninit(&pool);//删除线程池

}


int main (int argc, char *argv[])

{

thread_pool_demo();

return 0;

}

相关推荐

程序员:JDK的安装与配置(完整版)_jdk的安装方法

对于Java程序员来说,jdk是必不陌生的一个词。但怎么安装配置jdk,对新手来说确实头疼的一件事情。我这里以jdk10为例,详细的说明讲解了jdk的安装和配置,如果有不明白的小伙伴可以评论区留言哦下...

Linux中安装jdk并配置环境变量_linux jdk安装教程及环境变量配置

一、通过连接工具登录到Linux(我这里使用的Centos7.6版本)服务器连接工具有很多我就不一一介绍了今天使用比较常用的XShell工具登录成功如下:二、上传jdk安装包到Linux服务器jdk...

麒麟系统安装JAVA JDK教程_麒麟系统配置jdk

检查检查系统是否自带java在麒麟系统桌面空白处,右键“在终端打开”,打开shell对话框输入:java–version查看是否自带java及版本如图所示,系统自带OpenJDK,要先卸载自带JDK...

学习笔记-Linux JDK - 安装&amp;配置

前提条件#检查是否存在JDKrpm-qa|grepjava#删除现存JDKyum-yremovejava*安装OracleJDK不分系统#进入安装文件目...

Linux新手入门系列:Linux下jdk安装配置

本系列文章是把作者刚接触和学习Linux时候的实操记录分享出来,内容主要包括Linux入门的一些理论概念知识、Web程序、mysql数据库的简单安装部署,希望能够帮到一些初学者,少走一些弯路。注意:L...

测试员必备:Linux下安装JDK 1.8你必须知道的那些事

1.简介在Oracle收购Sun后,Java的一系列产品就被整合到Oracle官网中,打开官网乍眼一看也不知道去哪里下载,还得一个一个的摸索尝试,而且网上大多数都是一些Oracle收购Sun前,或者就...

Linux 下安装JDK17_linux 安装jdk1.8 yum

一、安装环境操作系统:JDK版本:17二、安装步骤第一步:下载安装包下载Linux环境下的jdk1.8,请去官网(https://www.oracle.com/java/technologies/do...

在Ubuntu系统中安装JDK 17并配置环境变量教程

在Ubuntu系统上安装JDK17并配置环境变量是Java开发环境搭建的重要步骤。JDK17是Oracle提供的长期支持版本,广泛用于开发Java应用程序。以下是详细的步骤,帮助你在Ubuntu系...

如何在 Linux 上安装 Java_linux安装java的步骤

在桌面上拥抱Java应用程序,然后在所有桌面上运行它们。--SethKenlon(作者)无论你运行的是哪种操作系统,通常都有几种安装应用程序的方法。有时你可能会在应用程序商店中找到一个应用程序...

Windows和Linux环境下的JDK安装教程

JavaDevelopmentKit(简称JDK),是Java开发的核心工具包,提供了Java应用程序的编译、运行和开发所需的各类工具和类库。它包括了JRE(JavaRuntimeEnviro...

linux安装jdk_linux安装jdk软连接

JDK是啥就不用多介绍了哈,外行的人也不会进来看我的博文。依然记得读大学那会,第一次实验课就是在机房安装jdk,编写HelloWorld程序。时光飞逝啊,一下过了十多年了,挣了不少钱,买了跑车,娶了富...

linux安装jdk,全局配置,不同用户不同jdk

jdk1.8安装包链接:https://pan.baidu.com/s/14qBrh6ZpLK04QS8ogCepwg提取码:09zs上传文件解压tar-zxvfjdk-8u152-linux-...

运维大神教你在linux下安装jdk8_linux安装jdk1.7

1.到官网下载适合自己机器的版本。楼主下载的是jdk-8u66-linux-i586.tar.gzhttp://www.oracle.com/technetwork/java/javase/downl...

window和linux安装JDK1.8_linux 安装jdk1.8.tar

Windows安装JDK1.8的步骤:步骤1:下载JDK打开浏览器,找到JDK下载页面https://d.injdk.cn/download/oraclejdk/8在页面中找到并点击“下载...

最全的linux下安装JavaJDK的教程(图文详解)不会安装你来打我?

默认已经有了linux服务器,且有root账号首先检查一下是否已经安装过java的jdk任意位置输入命令:whichjava像我这个已经安装过了,就会提示在哪个位置,你的肯定是找不到。一般我们在...

取消回复欢迎 发表评论: