信息发布→ 登录 注册 退出

C# 信号量(Semaphore)的应用 - 控制对资源的并发访问数

发布时间:2025-12-14

点击量:
应使用 SemaphoreSlim 控制并发许可数而非线程数,适用于限流场景;需用 try/finally 或 C#12+ using 确保 Release,避免许可泄露。

c# 信号量(semaphore)的应用 - 控制对资源的并发访问数

信号量(Semaphore)在 C# 中是用来限制同时访问某资源的线程数量的同步原语。它不像 lock 那样只允许一个线程进入,而是允许最多 N 个线程并发执行——这个 N 就是信号量的初始计数。

什么时候该用 Semaphore 而不是 lock 或 Mutex?

当你需要“最多 N 个线程能同时操作某资源”,而不是“只能 1 个”,就该考虑 Semaphore。比如:

  • 限制数据库连接池中同时活跃的连接数(避免打满 DB)
  • 控制对某个外部 API 的并发调用频率(防止被限流或封 IP)
  • 模拟有限硬件资源(如只有 3 台打印机,最多 3 个任务可同时打印)

SemaphoreSlim 是日常首选

推荐用 SemaphoreSlim(轻量级、支持异步、托管实现),而不是老式的 Semaphore(基于操作系统内核对象,开销大、不支持 async/await)。

基本用法:

var semaphore = new SemaphoreSlim(3); // 最多 3 个线程能通过
<p>// 进入临界区(阻塞或等待)
await semaphore.WaitAsync(); 
try
{
// 执行受控操作:如调用 API、写文件、处理任务...
}
finally
{
semaphore.Release(); // 必须释放,否则计数永远不增加
}

注意释放必须被执行

Release() 不会自动调用,必须确保它在任何路径下都执行——尤其是异常发生时。所以一定要包在 try/finally 或使用 using(C# 12+ 支持 SemaphoreSlimusing 语法糖):

Glarity Glarity

Glarity是一款免费开源的AI浏览器扩展,提供YouTube视频总结、网页摘要、写作工具等功能,支持免费的镜像翻译,电子邮件写作辅助,AI问答等功能。

Glarity 131 查看详情 Glarity
// C# 12+ 推荐写法(自动 Release)
await using (await semaphore.WaitAsync())
{
    // 执行操作
}

如果用的是旧版本 C#,就老老实实写 try/finally

别和线程数混淆:它是“许可数”,不是“线程数”

信号量管理的是“可用许可(permit)”数量,和线程本身无关。同一个线程可以多次 WaitAsync()(只要还有许可),也可以多次 Release()(但不能超过初始值,否则抛异常)。所以设计时要明确:

  • 每个业务逻辑单元消耗 1 个许可(最常见)
  • 是否允许重入(通常不建议,容易逻辑混乱)
  • 超时控制:WaitAsync(TimeSpan.FromSeconds(5)) 避免无限等待

基本上就这些。用对了,SemaphoreSlim 是控制并发水位的低调利器;用错了,可能死锁或许可泄露。关键是理解它管的是“许可”,不是“谁在用”。

以上就是C# 信号量(Semaphore)的应用 - 控制对资源的并发访问数的详细内容,更多请关注其它相关文章!


相关文章: 虫虫漫画精品漫画官网_虫虫漫画精品漫画官网进入精品漫画  c++项目目录结构应该如何组织_c++工程化项目结构规范  yy漫画网页版官方入口_yy漫画官网登录页面链接  微博网页版主页入口 微博官方网站免登录访问  三星GalaxyZFold5怎样在相册制作折叠屏分镜_iPhone三星GalaxyZFold5相册制作折叠屏分镜【创意编辑】  win11如何卸载Windows更新补丁 Win11解决更新导致系统不稳定的问题【修复】  AO3镜像入口大全 AO3网页版内容访问全集  蛙漫正版漫画平台入口_蛙漫免费阅读全站漫画资源  LINUX怎么安装MySQL_LINUX数据库安装配置教程  在J*a中如何实现对象克隆避免共享数据_对象克隆安全实践指南  响应式图片在网页设计中的正确实现方法  vivo手机参数配置怎么增强信号_vivo手机参数配置信号增强方法  凉拌黄瓜怎么拌更入味 凉拌黄瓜简单家常做法  在J*a中如何在J*a中使用异常机制记录错误日志_异常日志实践经验  12306选座怎么选到商务座_12306商务座选择与配置说明  Windows10怎么开启存储感知 Windows10系统设置自动清理临时文件释放C盘空间【教程】  win11 arm版怎么安装 M1/M2 Mac虚拟机安装ARM win11的方法  AI泡沫首次被“刺破”:GPU十年都无法存活!  XML中包含HTML标签导致解析错误? 正确嵌入非XML数据的两种方法  Lar*el Excel导入时生成自定义递增ID的策略与实践  优化Lar*el Docker镜像:Composer与PHP版本控制策略  C#中解析不规范的HTML为XML 常见的坑与解决办法  小红书商家版怎样在笔记嵌入商品卡路径_小红书商家版在笔记嵌入商品卡路径【挂载教程】  快速CSGO开箱网站指南 CSGO开箱平台推荐  SteamMachine定价或为699美元 大家想入手吗?  文本文档写html代码怎么运行_文本文档html代码运行步骤【教程】  Windows10怎么开启夜间模式 Windows10系统设置调整色温与亮度缓解夜间用眼疲劳【教程】  如何创建独立于主系统的J*a运行环境_隔离式环境搭建策略  写好的html代码怎么运行出来_运行写好的html代码方法【教程】  Pandas DataFrame:高效添加条件计算列  NRF24L01数据传输深度解析:解决大载荷接收异常与分包策略  网易大神账号申诉需要多久_网易大神账号申诉流程说明  淘宝支付提示失败如何解决 淘宝支付流程优化方法  怎么在html里运行vbs脚本_html中运行vbs脚本方法【教程】  Python多线程中正确使用sigwait处理SIGALRM信号  京东单号查询入口_京东快递订单追踪入口  C#使用XPath查询节点时出错? 常见语法错误与调试技巧  在Blazor WebAssembly应用中动态注入客户端特定指标代码的策略  中兴Axon42Ultra怎样在文件App筛图_iPhone中兴Axon42Ultra文件App筛图【图片筛选】  Spyder启动失败:字体文件权限拒绝错误解决方案  win11怎么查看应用耗电情况 Win11电池设置查看应用能耗排行榜【优化】  印象笔记如何设提醒任务防漏执行_印象笔记设提醒任务防漏执行【任务提醒】  微信网页版官方快速登录入口 微信网页版网页版账号直达  怎样使用“本地安全策略”提升Windows安全性_Secpol.msc配置指南【高手】  sublime怎么进行远程开发编辑_配置rsub/rmate实现sublime编辑服务器文件  Lar*el头像管理:图片缩放与旧文件删除的最佳实践  PHP表单提交后函数重复执行的解决方案:管理$_POST数据  谷歌浏览器无痕模式怎么开 Chrome开启无痕浏览设置方法【教程】  qq浏览器打开空白页怎么办 qq浏览器启动后显示白屏的解决教程  如何在离线环境中使用Composer_Composer离线安装依赖包的技巧与策略 

在线客服
服务热线

服务热线

4008988990

微信咨询
二维码
返回顶部
×二维码

截屏,微信识别二维码

打开微信

微信号已复制,请打开微信添加咨询详情!