C# ManualResetEvent‌的高级用法

server/2025/3/18 13:06:55/

一、ManualResetEvent 的核心作用‌

ManualResetEvent 是 C# 中用于 ‌线程同步‌ 的类(位于 System.Threading 命名空间),通过信号机制控制线程的等待与执行。其核心功能包括:

  • 阻塞线程‌:调用 WaitOne() 的线程会等待,直到事件被触发(信号状态)。
  • 手动控制信号‌:通过 Set() 发送信号释放所有等待线程,Reset() 重置为非信号状态。

‌二、基本用法‌

  1. ‌初始化‌
// 初始化时指定初始状态(true:已触发,false:未触发)
ManualResetEvent resetEvent = new ManualResetEvent(false);
  1. ‌阻塞线程(等待信号)‌
// 阻塞当前线程,直到收到信号或超时
bool signaled = resetEvent.WaitOne();              // 无限等待
bool signaled = resetEvent.WaitOne(3000);          // 等待3秒
bool signaled = resetEvent.WaitOne(TimeSpan.FromSeconds(3)); // 同上
  1. ‌发送信号‌
resetEvent.Set();   // 触发事件,释放所有等待线程
  1. ‌重置信号‌
resetEvent.Reset(); // 重置为非触发状态

‌三、典型场景‌

  1. ‌多线程任务协调‌
    多个线程等待某个操作完成后继续执行:
ManualResetEvent event = new ManualResetEvent(false);void ThreadWork()
{Console.WriteLine("线程等待中...");event.WaitOne();  // 阻塞直到事件触发Console.WriteLine("线程继续执行");
}// 启动多个线程
new Thread(ThreadWork).Start();
new Thread(ThreadWork).Start();Thread.Sleep(2000);
event.Set();  // 释放所有线程
  1. ‌异步操作完成通知‌
    主线程等待异步任务完成:
ManualResetEvent doneEvent = new ManualResetEvent(false);void AsyncOperation()
{Thread.Sleep(3000); // 模拟耗时操作doneEvent.Set();     // 标记完成
}new Thread(AsyncOperation).Start();
doneEvent.WaitOne();     // 主线程等待Console.WriteLine("异步操作完成");
  1. ‌超时控制‌
    限制操作的执行时间:
ManualResetEvent timeoutEvent = new ManualResetEvent(false);void LongRunningTask()
{Thread.Sleep(5000); // 模拟耗时操作timeoutEvent.Set();  // 完成后触发事件
}new Thread(LongRunningTask).Start();// 等待最多3秒
if (!timeoutEvent.WaitOne(3000))
{Console.WriteLine("操作超时");
}
else
{Console.WriteLine("操作正常完成");
}

‌四、与 AutoResetEvent 的区别‌

特性ManualResetEventAutoResetEvent
‌信号触发后状态‌保持触发状态,需手动重置自动重置为非触发状态
‌释放线程数‌释放所有等待线程仅释放一个等待线程
‌典型场景‌多线程同时启动、任务协调单次信号通知(如生产者-消费者)

‌五、最佳实践‌

‌使用 using 释放资源‌
实现 IDisposable,确保释放内核资源:

using (ManualResetEvent resetEvent = new ManualResetEvent(false))
{// 操作...
}

‌避免死锁‌

确保在所有可能的代码路径中调用 Set(),避免线程永久阻塞。
结合 try-finally 处理异常:

try
{// 代码...
}
finally
{resetEvent.Set();
}

‌超时设置‌
始终为 WaitOne() 指定合理的超时时间,防止线程无限等待:

if (!resetEvent.WaitOne(5000))
{throw new TimeoutException("等待超时");
}

‌替代方案‌
在 .NET 4.0+ 中,优先使用 Task 和 CancellationToken 实现异步控制:

var cts = new CancellationTokenSource();
Task.Run(() => LongMethod(cts.Token), cts.Token);if (!task.Wait(3000))
{cts.Cancel();Console.WriteLine("任务超时终止");
}

六、常见问题‌

‌Q:为什么调用 Set() 后线程未继续执行?‌

  • 原因‌:可能忘记调用 Reset(),导致后续 WaitOne() 直接通过。
  • ‌解决‌:在需要重新等待时调用 Reset()。
    ‌Q:多次调用 Set() 是否有副作用?‌
  • ‌答案‌:无。多次调用 Set() 等效于一次调用(事件保持触发状态)。

‌七、总结‌

  • ‌核心用途‌:实现多线程间的精确协调和同步。
  • ‌适用场景‌:需要手动控制信号状态的线程阻塞与释放(如批量任务启动、超时控制)。
  • 替代方案‌:在异步编程中,优先使用 Task、SemaphoreSlim 或 Barrier

http://www.ppmy.cn/server/175964.html

相关文章

【实测闭坑】LazyGraphRAG利用本地ollama提供Embedding model服务和火山引擎的deepseek API构建本地知识库

LazyGraphRAG 2024年4月,为解决传统RAG在全局性的查询总结任务上表现不佳,微软多部门联合提出Project GraphRAG(大模型驱动的KG);2024年7月,微软正式开源GraphRAG项目,引起极大关注&#xff0c…

【前端面试题】宏任务与微任务的区别

宏任务与微任务的区别 JavaScript采用单线程模型,通过 事件循环(Event Loop) 机制处理异步操作。 类比于厨师上菜的过程,顾客点的菜可能存在容易处理的 “软菜” 与难处理的 “硬菜” ,以及要加米饭酒水这些立马可以上…

Java常用设计模式

设计模式是软件开发中解决常见问题的模板或指南。Java中的23种设计模式通常被分为三大类:创建型模式(Creational Patterns)、结构型模式(Structural Patterns)和行为型模式(Behavioral Patterns&#xff09…

python脚本实现服务器内存和cpu使用监控,并记录日志,可以设置阈值和采样频率

Python 脚本,实现以下功能: 按日期自动生成日志文件(例如 cpu_mem_20231001.csv)当 CPU 或内存超过阈值时触发记录独立记录报警事件(保存到 alert.log)支持自定义阈值和监控间隔 脚本代码 import psutil …

HCIA-PPP

一、基本概念 1、定义:PPP 协议是一种数据链路层协议,在两点之间建立直接通信连接。常用于拨号上网、宽带接入、路由器间通信等。 2、核心功能: ①链路控制:建立、配置和测试数据链路连接。 ②网络层协议支持:支持…

【密码学——基础理论与应用】李子臣编著 第四章 SM4分组密码算法 课后习题

免责声明 这里都是自己搓或者手写的。 里面不少题目感觉有问题或者我的理解有偏颇,请大佬批评指正! 不带思考抄作业的请自动退出,我的并非全对,仅仅提供思维! SM4的python实现 基于AI生成的SM4加密算法-CSDN博客 题…

【云原生技术】容器技术的发展史

一、Jail 时代 容器不是一个新概念或者新技术,很早就有了,只是近几年遇到了云计算,整个技术 被彻底引爆了。 1.1 1979年 贝尔实验室发明 chroot chroot系统调用是在 1979 年开发第 7 版 Unix 期间引入的。贝尔实验室在 Unix V7 的 开发过程…

【STM32】USART串口协议串口外设-学习笔记

串口协议 通信接口 通信的目的:将一个设备的数据传送到另一个设备,扩展硬件系统。比如STM32芯片内部集成了很多功能模块,像定时器计数、PWM输出、AD采集等等。这些都是芯片内部的电路,这些电路的配置寄存器,数据寄存…