0

0

C# CountdownEvent使用方法 C#如何等待指定数量的信号

畫卷琴夢

畫卷琴夢

发布时间:2026-03-09 12:31:57

|

682人浏览过

|

来源于php中文网

原创

countdownevent 初始化必须指定正整数计数值,任务数应等于初始计数值;signal() 线程安全且需在 finally 中调用以防异常漏发;wait() 无 await 版本,不支持 async/await 原生等待。

c# countdownevent使用方法 c#如何等待指定数量的信号

CountdownEvent 初始化时必须指定初始计数值

创建 CountdownEvent 实例时,必须传入一个正整数作为初始计数,它代表“还需等待多少次 Signal() 才能触发完成”。传 0 会直接进入终止状态(IsSet == true),后续调用 Wait() 会立即返回,但容易让人误以为“还没开始就结束了”。

常见错误是把任务数量和计数值搞反:比如启动 5 个异步操作,却初始化为 new CountdownEvent(4),导致少等一次信号就提前唤醒;或初始化为 new CountdownEvent(6),最后一直阻塞超时。

  • 正确做法:任务数 = 初始计数值,例如 var cde = new CountdownEvent(5)
  • 若需动态调整(如中途取消某个任务),可用 cde.AddCount(1)cde.Signal(-1),但要注意线程安全和负值限制
  • 不建议复用已触发的 CountdownEvent,应新建实例

Wait() 阻塞等待直到计数归零,支持超时和取消

Wait() 是核心同步点,它会阻塞当前线程(或 async 等价物)直到内部计数变为 0。它不是轮询,底层基于内核事件对象,效率高。

务必注意:没有 await 版本的 Wait() —— CountdownEvent 是同步类型,不支持原生 async/await。若在 async 方法中使用,应包裹在 Task.Run(() => cde.Wait()) 中,但要警惕线程池开销;更推荐搭配 CancellationToken 使用以支持及时中断。

  • 带超时:cde.Wait(TimeSpan.FromSeconds(3)) 返回 false 表示超时未完成
  • 带取消:cde.Wait(ct),其中 ctCancellationToken,触发后抛 OperationCanceledException
  • 不要在 UI 线程直接调用 Wait()(尤其无超时),否则界面冻结

Signal() 减少计数,多线程调用安全

Signal() 是线程安全的原子操作,每次调用使内部计数减 1。多个线程可并发调用,无需额外锁。这是它比手动维护 int 计数 + lock 更可靠的关键原因。

典型场景是在每个异步任务结束时调用,例如 Task.Run(() => { DoWork(); cde.Signal(); })。但要注意:如果任务抛出异常未捕获,Signal() 就不会执行,导致计数永远不归零 —— 这是最常见的挂起原因。

  • 务必确保每个分支(包括 catchfinally)都调用 Signal(),或统一放在 finally 块里
  • 可传参 Signal(2) 一次性减去多个值,适用于一个任务代表多个子单元完成的情况
  • 调用 Signal() 超过初始计数值不会报错,但计数会变成负数(CurrentCount 可为负),IsSet 仍为 true

CountdownEvent 不适合替代 Task.WhenAll 或 ManualResetEventSlim

它解决的是“等待 N 次独立信号”的问题,不是“等待所有 Task 完成”或“单次广播通知”。混淆用途会导致代码难维护甚至死锁。

例如,用 CountdownEvent 等待 10 个 Task,却在 Task.ContinueWith 里调用 Signal(),若 ContinueWith 调度失败或被取消,信号就丢失;而 Task.WhenAll(tasks).Wait() 是语义明确、自带错误传播的方案。

  • 优先用 Task.WhenAll 处理已知数量的 Task
  • ManualResetEventSlim 实现“单次唤醒多个等待者”
  • CountdownEvent 的真实优势场景:工作项由不同线程/模块动态注册并完成(如 IOCP 回调、事件处理器、插件系统),无法预建 Task 列表
实际使用中最容易被忽略的是异常路径下漏掉 Signal(),以及误以为它支持 async 等待。这两个点一旦出错,调试时往往表现为“程序卡住不动”,且无明显异常堆栈。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

990

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

607

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

314

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

438

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

601

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

438

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

601

2023.08.10

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

59

2026.03.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.9万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号