linux之线程池

1. 简介

• 线程池是一种用于管理和重用多个线程的设计模式 。它通过维护一个线程池(线程的集合),可以有效地处理并发任务而无需每次都创建和销毁线程 。这种方法可以减少线程创建和销毁的开销,提高性能和资源利用率。 提高响应速度,任务来了直接交给空闲线程执行,无需等待线程创建。

2. Glib的工作流程

• 线程池创建:首先创建一个线程池,指定任务函数和其他参数 。线程池会创建一定数量的线程,这些线程进入等待状态,准备执行任务,或在提交任务后才创建线程(取决于配置)。线程池中的所有线程执行的都是同一个任务函数。

• 任务队列:线程池维护一个任务队列。当我们向线程池提交任务时,任务会被放入这个队列中实际上,放入任务队列 的是 我们在提交任务时传递的任务数据。

• 线程执行任务:线程池中的线程从任务队列中取出任务数据,然后调用任务函数,执行任务。执行完成后,线程不会退出,而是继续从任务队列中取下一个任务执行。如果没有待执行的任务,线程通常在等待一段时间后被回收(取决于具体的配置)。

3. 相关函数

3.1 GFunc

cpp 复制代码
// 此处的 data 是在启动任务时,传递给每个任务的,而 user_data 是在创建线程池时传入的共享数据,对于每个任务都是一样的 
typedef void (*GFunc)(gpointer data, gpointer user_data);

3.2 g_thread_pool_new

cpp 复制代码
/** 
* @brief 创建新的线程池 
* 
* @param func 池中线程执行的函数 
* @param user_data 传递给 func 的数据,可以为 NULL,这里的 user_data 最终会被存储在 GThreadPool 结构体的 user_data 属性中 
* @param max_threads 线程池容量,即当前线程池中可以同时运行的线程数。-1 表示没有限制 
* @param exclusive 独占标记位。决定当前的线程池独占所有的线程 还是与其它线程池共享这些线程。取值可以是 TRUE 或 FALSE 
* TRUE:立即启动数量为 max_threads 的线程,且启动的线程只能被当前线程池使用 
* FALSE:只有在需要时,即需要执行任务时才创建线程,且线程可以被多个非独享资源的线程池共用 
* @param error 用于报告错误信息,可以是 NULL,表示忽略错误 
* @return GThreadPool* 线程池实例指针。无论是否发生错误,都会返回有效的线程池 
*/ 
GThreadPool *g_thread_pool_new( 
	GFunc func, 
	gpointer user_data, 
	gint max_threads, 
	gboolean exclusive, 
GError **error);

3.3 g_thread_pool_push

cpp 复制代码
/** 
* @brief 向 pool 指向的线程池实例添加数据,这一行为实际上会向任务队列添加新的任务。当存在可用线程时任务立即执行,否则任务数据会一直待在队列中,直至腾出可用线程执行任务 
* 
* @param pool 指向线程池实例的指针 
* @param data 传递给每个任务的独享数据 
* @param error 错误信息 
* @return gboolean 成功返回 TRUE,失败返回 FALSE 
*/ 
gboolean g_thread_pool_push( 
	GThreadPool *pool, 
	gpointer data, 
GError **error);

3.4 g_thread_pool_free

cpp 复制代码
/** 
* @brief 释放为 pool 指向的线程池分配的所有资源 
* 
* @param pool 线程池指针 
* @param immediate 是否立即释放线程池 
* TRUE:立即释放所有资源,未处理的数据不被处理 
* FALSE:在最后一个任务执行完毕之前,线程池不会被释放 
* 需要注意的是:执行任务时,线程池的任何一个线程都不会被打断。无论这个参数是何取值,都可以保证至少线程池释放前正在运行的线程可以完成它们的任务。 
* @param wait_ 当前函数是否阻塞等待 所有任务完成 
* TRUE:所有需要处理的任务执行完毕当前函数才会返回 
* FALSE:当前函数立即返回 
*/ 
void g_thread_pool_free ( 
	GThreadPool* pool, 
	gboolean immediate, 
	gboolean wait_ 
);

3.5 例子

cpp 复制代码
#include <glib.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>


void task(gpointer data,gpointer user_data){
    int tmp = *((int *)data);//每个任务独享的数据。
    free(data);
    printf("任务%d开始执行\n",tmp);
    sleep(tmp);
    printf("任务%d执行结束\n",tmp);
}

int main(int argc, char const *argv[])
{
    //创建线程池
    GThreadPool * gtp = g_thread_pool_new(task,NULL,5,TRUE,NULL);

    //提交任务10个
    for (size_t i = 0; i < 10; i++)
    {
        int * tmp = malloc(sizeof(int));
        *tmp =  i + 1;
        g_thread_pool_push(gtp,tmp,NULL);
    }
    
    //释放线程池
    g_thread_pool_free(gtp,FALSE,TRUE);
    printf("所有任务都执行完成了\n");
    return 0;
}

• 结果,如图:

相关推荐
ZFB000112 小时前
【麒麟桌面系统】V10-SP1 2503 系统知识——插入U盘(移动硬盘)为只读状态
linux·运维·kylin
unfeeling_13 小时前
Keepalived实验
linux·服务器·网络
Web极客码13 小时前
解决WordPress后台“外观”菜单消失
linux·服务器·wordpress
熬夜有啥好13 小时前
Linux软件编程——综合小练习
linux·算法·目录遍历·fgets·strcpy·linux内核与用户交互·strtok
qizhideyu13 小时前
LVS(Linux virual server)
linux·运维·lvs
xiaoliuliu1234513 小时前
CentOS 7 安装 gcc-4.8.5-44.el7.x86_64.rpm 详细步骤(含依赖解决)
linux·运维·centos
青桔柠薯片13 小时前
Linux软件编程:线程和进程间通信
linux·开发语言·线程·进程
在这habit之下13 小时前
Linux Virtual Server(LVS)学习总结
linux·学习·lvs
feng68_14 小时前
Nginx高性能Web服务器
linux·运维·服务器·nginx
wefg114 小时前
【Linux】进程的页表详解
linux