线程池模式与C#中用法

一、线程池模式解析

1. 核心概念

线程池是一种 管理线程生命周期的技术,主要解决以下问题:

  • 减少线程创建/销毁开销:复用已存在的线程

  • 控制并发度:避免无限制创建线程导致资源耗尽

  • 任务队列:有序处理异步请求

2. 工作流程

 

3. .NET 中的实现
  • 内置线程池System.Threading.ThreadPool

  • 关键方法

ThreadPool.QueueUserWorkItem(state => { /* 任务逻辑 */ });

4. 线程池组成System.Threading.ThreadPool
组件说明
Worker Threads处理普通任务
I/O Completion Ports处理异步I/O操作
任务队列存放等待执行的任务

二、基本使用方法

1. 提交任务
// 使用 QueueUserWorkItem
ThreadPool.QueueUserWorkItem(state => 
{Console.WriteLine($"Task executed on thread {Thread.CurrentThread.ManagedThreadId}");// 长时间运行的任务
});// 带参数的任务
ThreadPool.QueueUserWorkItem(obj => 
{var data = (string)obj;Console.WriteLine($"Processing: {data}");
}, "Hello ThreadPool");

2. 获取线程池状态

ThreadPool.GetAvailableThreads(out int workerThreads, out int completionPortThreads);
Console.WriteLine($"可用工作线程: {workerThreads}, I/O线程: {completionPortThreads}");ThreadPool.GetMinThreads(out int minWorker, out int minIO);
ThreadPool.GetMaxThreads(out int maxWorker, out int maxIO);

 

三、高级配置

1. 设置线程数限制
// 设置最小线程数(预热)
ThreadPool.SetMinThreads(4, 4);// 设置最大线程数(默认值通常足够)
ThreadPool.SetMaxThreads(16, 16); // 不推荐随意修改
2. 使用 Task 封装(现代推荐方式)
Task.Run(() => 
{// 会自动使用ThreadPoolConsole.WriteLine("Running via Task");
});
3. 带返回值的任务
var result = await Task.Run(() => 
{Thread.Sleep(1000);return 42;
});
Console.WriteLine($"Result: {result}");

四、最佳实践

1. 适合场景
场景示例
短期任务<1秒完成的任务
I/O密集型文件/网络操作
并行计算简单的数据分块处理
2. 不适合场景
场景问题替代方案
长时间运行阻塞线程池线程new Thread() 或 LongRunning 任务
需要优先级线程池无优先级自定义优先级队列
精细控制需要线程亲和性专用线程
3. 性能调优建议
// 在应用启动时预热线程池
ThreadPool.SetMinThreads(Environment.ProcessorCount * 2, Environment.ProcessorCount * 2);// 监控线程池状态
PerformanceCounter poolCounter = new PerformanceCounter("ThreadPool", "Thread Count", Process.GetCurrentProcess().ProcessName);

五、与 DelegateSpooler 对比

特性ThreadPoolDelegateSpooler
线程管理自动手动控制
任务队列全局共享独立实例
优先级不支持可自定义
适用场景通用短期任务需要特殊控制的场景

 

六、完整示例

1. 批量处理任务
using System;
using System.Threading;class Program
{static void Main(){// 设置最小线程数ThreadPool.SetMinThreads(4, 4);// 提交10个任务for (int i = 0; i < 10; i++){int taskId = i;ThreadPool.QueueUserWorkItem(_ => {Console.WriteLine($"Task {taskId} started on thread {Thread.CurrentThread.ManagedThreadId}");Thread.Sleep(1000); // 模拟工作Console.WriteLine($"Task {taskId} completed");});}Console.ReadLine();}
}
2. 异步I/O操作
using System.Net;
using System.IO;ThreadPool.QueueUserWorkItem(_ => 
{var request = WebRequest.Create("https://example.com");using var response = request.GetResponse();using var reader = new StreamReader(response.GetResponseStream());Console.WriteLine(reader.ReadToEnd());
});

七、常见问题解决

1. 线程饥饿

现象:任务长时间排队不执行
解决

// 增加最小线程数
ThreadPool.SetMinThreads(Environment.ProcessorCount * 2, Environment.ProcessorCount * 2);
2. 异常处理
ThreadPool.QueueUserWorkItem(_ => 
{try{// 可能抛出异常的代码}catch (Exception ex){Console.WriteLine($"Task failed: {ex}");}
});
3. 取消任务
var cts = new CancellationTokenSource();// 提交可取消任务
Task.Run(() => 
{while (!cts.IsCancellationRequested){// 工作代码}
}, cts.Token);// 取消所有任务
cts.Cancel();

总结

  • 简单任务:优先使用 ThreadPool.QueueUserWorkItem

  • 现代开发:推荐使用 Task.Run(内部使用线程池)

  • 复杂场景:考虑自定义线程池(如 DelegateSpooler

  • 关键原则:避免阻塞线程池线程,保持任务短小精悍

通过合理使用线程池,可以显著提升应用程序的并发性能和资源利用率。

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

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

相关文章

设置IDEA打开新项目使用JDK17

由于最近在学习Spring-AI&#xff0c;所以JDK8已经不适用了&#xff0c;但是每次创建新项目都还是JDK8&#xff0c;每次调来调去很麻烦 把Projects和SDKs都调整为JDK17即可 同时&#xff0c;Maven也要做些更改&#xff0c;主要是添加build标签 <build><plugins>&…

初识MySQL · 索引

目录 前言&#xff1a; 重温磁盘 认识索引 为什么这么做&#xff0c;怎么做 重谈page 聚簇索引VS非聚簇索引 回表查询 索引分类 前言&#xff1a; 前文我们主要是介绍了MySQL的一些基本操作&#xff0c;增删查改一类的操作都介绍了&#xff0c;并且因为大多数情况下&am…

MySQL——7、复合查询和表的内外连接

复合查询和表的内外连接 1、基本查询回顾2、多表查询3、自连接4、子查询4.1、单行子查询4.2、多行子查询4.3、多列子查询4.4、在from子句中使用子查询4.5、合并查询 5、表的内连和外连5.1、内连接5.2、外连接5.2.1、左外连接5.2.2、右外连接 1、基本查询回顾 1.1、查询工资高于…

MYSQL故障排查和环境优化

一、MySQL故障排查 1. 单实例常见故障 &#xff08;1&#xff09;连接失败类问题 ERROR 2002 (HY000): Cant connect to MySQL server 原因&#xff1a;MySQL未启动或端口被防火墙拦截。 解决&#xff1a;启动MySQL服务&#xff08;systemctl start mysqld&#xff09;或开放…

7GB显存如何部署bf16精度的DeepSeek-R1 70B大模型?

构建RAG混合开发---PythonAIJavaEEVue.js前端的实践-CSDN博客 服务容错治理框架resilience4j&sentinel基础应用---微服务的限流/熔断/降级解决方案-CSDN博客 conda管理python环境-CSDN博客 快速搭建对象存储服务 - Minio&#xff0c;并解决临时地址暴露ip、短链接请求改…

数字图像处理——图像压缩

背景 图像压缩是一种减少图像文件大小的技术&#xff0c;旨在在保持视觉质量的同时降低存储和传输成本。随着数字图像的广泛应用&#xff0c;图像压缩在多个领域如互联网、移动通信、医学影像和卫星图像处理中变得至关重要。 技术总览 当下图像压缩JPEG几乎一统天下&#xff…

抖音视频怎么去掉抖音号水印

你是不是经常遇到这样的烦恼&#xff1f;看到喜欢的抖音视频&#xff0c;想保存下来分享给朋友或二次创作&#xff0c;却被抖音号水印挡住了画面&#xff1f;别着急&#xff0c;今天教你几种超简单的方法&#xff0c;轻松去除水印&#xff0c;高清无水印视频一键保存&#xff0…

RISC-V 开发板 MUSE Pi Pro PCIE 测试以及 fio 崩溃问题解决

视频讲解&#xff1a; RISC-V 开发板 MUSE Pi Pro PCIE 测试以及 fio 崩溃问题解决 板子上有一个m.2的pcie插槽&#xff0c;k1有三个pcie控制器&#xff0c;pcie0和usb3复用一个phy&#xff0c;所以实际开发板就两个&#xff0c;测试的话&#xff0c;上一个nvme硬盘&#xff0c…

超级管理员租户资源初始化与授权管理设计方案

背景说明 在多租户系统中&#xff0c;资源&#xff08;如功能模块、系统菜单、服务能力等&#xff09;需按租户维度进行授权管理。超级管理员在创建新租户时&#xff0c;需要初始化该租户的资源授权信息。 两种可选方案 方案描述方案 A&#xff1a;前端传入选中的资源列表创…

stm32week16

stm32学习 十一.中断 4.使用中断 EXTI的配置步骤&#xff1a; 使能GPIO时钟设置GPIO输入模式使能AFIO/SYSCFG时钟设置EXTI和IO对应关系设置EXTI屏蔽&#xff0c;上/下沿设置NVIC设计中断服务函数 HAL库的使用&#xff1a; 使能GPIO时钟&#xff1a;__HAL_RCC_GPIOx_CLK_EN…

什么是RDMA?

什么是RDMA&#xff1f; RDMA(RemoteDirect Memory Access)技术全称远程直接内存访问&#xff0c;就是为了解决网络传输中服务器端数据处理的延迟而产生的。它将数据直接从一台计算机的内存传输到另一台计算机&#xff0c;无需双方操作系统的介入。这允许高吞吐、低延迟的网络…

golang 安装gin包、创建路由基本总结

文章目录 一、安装gin包和热加载包二、路由简单场景总结 一、安装gin包和热加载包 首先终端新建一个main.go然后go mod init ‘项目名称’执行以下命令 安装gin包 go get -u github.com/gin-gonic/gin终端安装热加载包 go get github.com/pilu/fresh终端输入fresh 运行 &…

【数据结构篇】链式结构二叉树

目录&#xff1a; 一 二叉链的概念与结构&#xff1a; 1.1 概念&#xff1a; 1.2 结构&#xff1a; 二 二叉链的实现&#xff1a; 2.1 二叉树的构建&#xff1a; 2.2 二叉树的遍历&#xff1a; 2.2.1 前序遍历&#xff1a; 2.2.2 中序遍历&#xff1a; 2.2.3 后序遍历…

【MySQL】02.数据库基础

1. 数据库的引入 之前存储数据用文件就可以了&#xff0c;为什么还要弄个数据库? 文件存储存在安全性问题&#xff0c;文件不利于数据查询和管理&#xff0c;文件不利于存储海量数据&#xff0c;文件在程序中控制不方便。而为了解决上述问题&#xff0c;专家们设计出更加利于…

什么是 Langchain 以及其核心组件

LangChain 官方文档&#xff1a;LangChain 一、什么是Langchain LangChain 是一个用于构建基于LLM的应用框架&#xff0c;它提供了对 LLM API 的封装和扩展&#xff0c;使开发者能够更方便地构建复杂的应用。 个人理解&#xff1a;用类比的方法来说&#xff0c;LangChain类似…

博客系统功能测试

博客系统网址&#xff1a;http://8.137.19.140:9090/blog_list.html 主要测试内容 功能测试、界面测试、性能测试、易用性测试、安全测试、兼容性测试、弱网测试、安装卸载测试、压力测试… 测试方法及目的 利用selenium和python编写测试脚本&#xff0c;对博客系统进行的相关…

项目制作流程

一、使用 CRA 创建项目 npx create-react-app name 二、按照业务规范整理项目目录 &#xff08;重点src目录&#xff09; 三、安装插件 npm install sass -Dnpm install antd --savenpm install react-router-dom 四、配置基础路由 Router 1. 安装路由包 react-router-dom …

ngx_http_random_index_module 模块概述

一、使用场景 随机内容分发 当同一目录下存放多份等价内容&#xff08;如多张轮播图、不同版本静态页面等&#xff09;时&#xff0c;可通过随机索引实现负载均衡或流量分散。A/B 测试 通过目录请求自动随机分配用户到不同测试组&#xff0c;无需后端逻辑参与。动态“首页”选…

智能权限守护者:基于Python描述符的动态角色控制实现

智能权限守护者:基于Python描述符的动态角色控制实现 引言:当描述符遇见权限管理 在Python的魔法方法体系中,描述符(Descriptor)以其优雅的属性访问控制机制著称。当我们将描述符与RBAC(基于角色的访问控制)模型结合,就能创造出既灵活又安全的动态权限管理系统。本文…

Linux 的 UDP 网络编程 -- 回显服务器,翻译服务器

目录 1. 回显服务器 -- echo server 1.1 相关函数介绍 1.1.1 socket() 1.1.2 bind() 1.1.3 recvfrom() 1.1.4 sendto() 1.1.5 inet_ntoa() 1.1.6 inet_addr() 1.2 Udp 服务端的封装 -- UdpServer.hpp 1.3 服务端代码 -- UdpServer.cc 1.4 客户端代码 -- UdpClient.…