【C语言实现线程池】

创建一个线程池是提高多线程应用程序性能的有效方法。一个线程池中包含一定数量的工作线程,这些线程可以复用来处理多个任务,避免了频繁创建和销毁线程所带来的开销。

下面是一个基础的线程池实现的框架,使用C语言和POSIX线程库(pthread)。这个简单的线程池会轮询并执行放入队列的任务。

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <unistd.h>#define MAX_THREADS 4
#define MAX_QUEUE 8// 任务结构体
typedef struct {void (*function)(void *); // 任务的回调函数void *argument;          // 回调函数的参数
} threadpool_task_t;// 线程池结构体
typedef struct {pthread_mutex_t lock;       // 互斥锁pthread_cond_t notify;      // 条件变量pthread_t *threads;         // 线程数组threadpool_task_t *queue;   // 任务队列int thread_count;           int queue_size;             int head;                   int tail;                   int count;                  // 当前任务数量int shutdown;               // 是否关闭线程池int started;                // 已启动的线程数量
} threadpool_t;// 线程工作函数
void *threadpool_thread(void *threadpool);// 创建线程池
threadpool_t *threadpool_create(int thread_count, int queue_size);// 向线程池中添加任务
int threadpool_add(threadpool_t *pool, void (*function)(void *), void *argument);// 销毁线程池
int threadpool_destroy(threadpool_t *pool, int flags);int main() {threadpool_t *pool = threadpool_create(MAX_THREADS, MAX_QUEUE);// 添加一些示例任务for (int i = 0; i < 10; i++) {threadpool_add(pool, &my_task, (void *)(long)i);}sleep(3); // 等待任务完成threadpool_destroy(pool, 0);return 0;
}// 任务示例
void my_task(void *arg) {printf("Thread %ld working on task %ld\n", pthread_self(), (long)arg);
}threadpool_t *threadpool_create(int thread_count, int queue_size) {threadpool_t *pool = (threadpool_t *)malloc(sizeof(threadpool_t));if (pool == NULL) {return NULL;}pool->thread_count = 0;pool->queue_size = queue_size;pool->head = pool->tail = pool->count = 0;pool->shutdown = pool->started = 0;pool->threads = (pthread_t *)malloc(sizeof(pthread_t) * thread_count);pool->queue = (threadpool_task_t *)malloc(sizeof(threadpool_task_t) * queue_size);pthread_mutex_init(&(pool->lock), NULL);pthread_cond_init(&(pool->notify), NULL);for (int i = 0; i < thread_count; i++) {if (pthread_create(&(pool->threads[i]), NULL, threadpool_thread, (void *)pool) != 0) {threadpool_destroy(pool, 0);return NULL;}pool->thread_count++;pool->started++;}return pool;
}int threadpool_add(threadpool_t *pool, void (*function)(void *), void *argument) {int next, err = 0;if (pthread_mutex_lock(&(pool->lock)) != 0) {return -1;}next = (pool->tail + 1) % pool->queue_size;do {if (pool->count == pool->queue_size) {err = 1;break;}pool->queue[pool->tail].function = function;pool->queue[pool->tail].argument = argument;pool->tail = next;pool->count += 1;if (pthread_cond_signal(&(pool->notify)) != 0) {err = 1;break;}} while (0);if (pthread_mutex_unlock(&(pool->lock)) != 0) {err = 1;}return err;
}int threadpool_destroy(threadpool_t *pool, int flags) {if (pool == NULL) {return -1;}if (pthread_mutex_lock(&(pool->lock)) != 0) {return -1;}pool->shutdown = 1;if ((pthread_cond_broadcast(&(pool->notify)) != 0) || (pthread_mutex_unlock(&(pool->lock)) != 0)) {return -1;}for (int i = 0; i < pool->thread_count; i++) {if (pthread_join(pool->threads[i], NULL) != 0) {return -1;}}if (pthread_mutex_destroy(&(pool->lock)) != 0 || pthread_cond_destroy(&(pool->notify)) != 0) {return -1;}free(pool->threads);free(pool->queue);free(pool);return 0;
}void *threadpool_thread(void *threadpool) {threadpool_t *pool = (threadpool_t *)threadpool;threadpool_task_t task;while (1) {pthread_mutex_lock(&(pool->lock));while ((pool->count == 0) && (!pool->shutdown)) {pthread_cond_wait(&(pool->notify), &(pool->lock));}if ((pool->shutdown) && (pool->count == 0)) {break;}task.function = pool->queue[pool->head].function;task.argument = pool->queue[pool->head].argument;pool->head = (pool->head + 1) % pool->queue_size;pool->count -= 1;pthread_mutex_unlock(&(pool->lock));(*(task.function))(task.argument);}pool->started--;pthread_mutex_unlock(&(pool->lock));pthread_exit(NULL);return NULL;
}

代码说明

  1. 结构体定义:

    • threadpool_task_t: 表示一个任务,包括任务的回调函数和参数。
    • threadpool_t: 表示线程池,包括线程数组、任务队列、队列头尾索引、互斥锁、条件变量等。
  2. 线程池工作函数threadpool_thread:

    • 线程从任务队列中获取任务并执行。
    • 在任务队列为空时,该线程将会阻塞等待新的任务。
  3. 线程池创建函数threadpool_create:

    • 初始化线程池的结构体,各种初始值和资源分配。
    • 创建指定数量的线程,并将它们与threadpool_thread函数关联起来。
  4. 添加任务函数threadpool_add:

    • 将任务放入队列,并通知一个阻塞在条件变量上的线程去执行这个任务。
  5. 销毁线程池函数threadpool_destroy:

    • 关闭线程池,等待所有线程终止,并销毁分配的资源。
  6. 示例任务my_task:

    • 简单输出线程ID和任务ID示例。

这个基础代码可以满足大部分简单的线程池需求,但在实际应用中,你可能还会需要考虑:

  • 线程池中的异常处理。
  • 动态调整线程池大小。
  • 更复杂的任务调度和优先级。
  • 更高效的任务队列实现。

这个基础代码是一个很好的起点,你可以在其基础上根据具体需求做进一步优化和扩展。

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

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

相关文章

lodash已死?radash库方法介绍及源码解析 —— 对象方法篇

assign&#xff1a;递归合并两个对象 使用说明 功能说明&#xff1a;类似于 JavaScript 的 Object.assign 方法&#xff0c;用于将 override 对象的属性和值复制到 initial 对象中。如果属性值是对象&#xff0c;则递归地进行赋值。 参数&#xff1a;初始对象、覆盖对象。 返…

清理mysql binglog文件

mysql随着使用时间的推移&#xff0c;binglog文件会越来越大&#xff0c;比如我们的oa系统&#xff0c;上线4年多了&#xff0c;最近总有磁盘空间满影响系统正常使用的情况出现。检查后发现binglog是罪归祸首。 binglog文件最好不要采用应删除的方式清理&#xff0c;如下方式可…

LLVM Visual Studio构建

cd llvm-project-main cmake -S llvm -B build -G "Visual Studio 16 2019" -DLLVM_ENABLE_PROJECTSclang-tools-extra -DLLVM_ENABLE_PROJECTSclang .

spring cache(三)demo

一、入门demo 1、pom <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.o…

ACL组网实验(华为)

思科设备参考&#xff1a;ACL组网实验&#xff08;思科&#xff09; 更多内容参考&#xff1a;华为ACL配置&#xff08;基本ACL高级ACL综合应用&#xff09; 技术简介 ACL&#xff08;Access Control List&#xff09;技术是一种基于包过滤的流控制技术&#xff0c;主要用于…

ML307R OpenCPU UART使用

一、串口使用流程图 二、串口相关函数介绍 三、实现串口收发 一、串口使用流程图 OneMO ML307R模组提供了2路UART给开发者用于通讯开发&#xff0c;以及1路DBG UART用于log的打印。UART Demo示例可以在SDK&#xff1a;examples\uart\src\cm_demo_uart.c中查看。 串口使用流…

Vue3实战笔记(42)—Vue + ECharts:流量数据可视化的强大组合

文章目录 前言vue3使用echarts标准demo&#xff1a;总结 前言 在前端开发中&#xff0c;数据可视化已经成为了一个不可或缺的部分。Vue.js作为一个轻量级且易于上手的渐进式JavaScript框架&#xff0c;与ECharts这个强大的数据可视化库的结合&#xff0c;使得在Vue应用中构建交…

windows下nginx配置https证书

1、制作证书 1.1 安装工具openSSL。下载地址&#xff1a;http://slproweb.com/products/Win32OpenSSL.html Win64OpenSSL_Light-3_1_0.exe安装&#xff08;假定安装位置在 d:\openSSL\&#xff09; 1.2 配置openSSL环境。 新建系统变量OpenSSL值为d:\openSSL\bin&#xff0c;相…

java继承(构造器)使用细节3

那么我们怎么来选择用父类的 有参构造器 和无参构造器泥&#xff1f; 就可以不写 或只写super() 并且如像爸爸的爸爸还有爸爸&#xff0c;还有构造。所以会一直调用到爷爷的构造。 细节2 直接继承指的是儿子和爸爸&#xff0c;爸爸和爷爷

Python 渗透测试:Redis 数据库 弱密码测试.(6379端口)

什么是 Redis 数据库 Redis (Remote Dictionary Server) 是一个开源的、内存中的数据结构存储系统&#xff0c;它可以用作数据库、缓存和消息中间件。它支持多种类型的数据结构,如字符串(strings)、哈希(hashes)、列表(lists)、集合(sets)、有序集合(sorted sets)等&#xff0…

View function mapping is overwriting an existing endpoint function: home_page

这个错误是因为在你的 Flask 应用中,你定义了两个或多个视图函数,它们使用了相同的 endpoint 名称。Flask 不允许多个视图函数使用相同的 endpoint 名称,因为这会导致冲突。 通常,这个错误会出现在以下几种情况下: 你在不同的路由装饰器中使用了相同的 endpoint 名称。你…

Xilinx(AMD) FPGA通过ICAP原语读取芯片IDCODE实现方法

1 概述 Xilinx每种型号的FPGA芯片都有一个唯一的IDCODE与之对应&#xff0c;同一型号不同封装的IDCODE是相同的。IDCODE的获取方法包括JTAG、ICAP原语、AXI_HWICAP IP核等。获取IDCODE常用于根据芯片型号改变代码的功能&#xff0c;或者对代码进行授权保护&#xff0c;只能在指…

Java使用apache.poi生成excel插入word中

加油&#xff0c;新时代打工人&#xff01; 工作需求&#xff0c;上个文章我们生成好的word&#xff0c;这次将生成好的excel表格数据&#xff0c;插入word中。需要准备好excle数据&#xff0c;然后插入到word中。 最后个需要&#xff0c;就是把这些生成好的word文档转成pdf进行…

【LeetCode面试经典150题】104. 二叉树的最大深度

一、题目 104. 二叉树的最大深度 - 力扣&#xff08;LeetCode&#xff09; 给定一个二叉树 root &#xff0c;返回其最大深度。 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。 二、思路 二叉树的题目就是递归&#xff0c;怎么递归呢&#xff1f;要求…

EyeMock下载与使用教程

视频眼神修复直视镜头的AI具有极高的灵活性和适应性。它可以根据不同的拍摄环境和主播需求进行个性化设置&#xff0c;确保最佳的视觉呈现效果。在直播互动中&#xff0c;主播可能因为分神或疲劳而失去与观众的直视&#xff0c;这款工具能够迅速识别并修复这一问题&#xff0c;…

机器学习(七) ----------聚类(K-means)

目录 1 核心思想 2 K-means算法 2.1 算法概述 2.2 算法步骤 2.3 数学原理 2.4 ‘肘’方法确定K值 2.4.1 原理 2.4.2 步骤 2.4.3 代码实现 2.5 聚类评估方法 2.5.1 SC轮廓系数&#xff08;Silhouette Coefficient&#xff09; 计算方法 解读 注意事项 2.5.2 Cal…

刷代码随想录有感(77):回溯算法——含有重复元素的全排列

题干&#xff1a; 代码&#xff1a; class Solution { public:vector<int> tmp;vector<vector<int>> res;void backtracking(vector<int> nums, vector<int> used){if(tmp.size() nums.size()){res.push_back(tmp);return;}sort(nums.begin(),…

ubuntu22部署 vue 和 springboot项目

vue前端系统部署&#xff1a; 1.使用命令npm run build:prod&#xff08;构建生产环境&#xff09;进行打包&#xff0c;打包成功后&#xff0c;会在项目下看到dist文件夹 ps:也有可能是npm run build命令进行打包(具体看package.json中"build": "vue-cli-serv…

对安卓手机上损坏的 SD 卡进行故障排除:恢复提示和修复

概括 如果您总是在旅途中&#xff0c;那么您很可能每天都在使用 SD 卡。这些微小但功能强大的闪存已经变得和手机的内部存储一样有用。它们可以存储数据并移动您想要的任何数据类型&#xff0c;因为它们在 Android 设备上添加了额外的存储空间。不幸的是&#xff0c;他们可能会…