C# WaitHandle类的几个有用的函数

WaitHandle类还提供静态方法来处理更复杂的同步难题。WaitAny、WaitAll和SignalAndWait方法可对多个句柄执行信号通知和等待操作,这些等待句柄可以是不同类型(包括Mutex和Semaphore,因为它们同样继承自抽象的WaitHandle类)。ManualResetEventSlim和CountdownEvent也能通过它们的WaitHandle属性参与这些方法。

1.WaitAny

WaitHandle.WaitAny 是 .NET 中用于多线程同步的重要方法,它允许线程等待多个同步对象中的任意一个变为信号状态。

函数定义:

cs 复制代码
public static int WaitAny(WaitHandle[] waitHandles);
public static int WaitAny(WaitHandle[] waitHandles, int millisecondsTimeout);
public static int WaitAny(WaitHandle[] waitHandles, TimeSpan timeout);
public static int WaitAny(WaitHandle[] waitHandles, int millisecondsTimeout, bool exitContext);

核心功能

  1. 多等待对象支持:可以同时监视多个等待句柄(如 ManualResetEvent、AutoResetEvent、Mutex、Semaphore 等)

  2. 任意触发机制:只要数组中任何一个等待句柄收到信号,等待就会结束

  3. 超时控制:可以设置超时时间,避免无限期等待

  4. 返回值指示:返回触发等待的句柄在数组中的索引

下面给一个简单的例子:

cs 复制代码
// 创建三个等待句柄
var waitHandle1 = new AutoResetEvent(false);
var waitHandle2 = new ManualResetEvent(false);
var waitHandle3 = new Semaphore(0, 1);

WaitHandle[] handles = { waitHandle1, waitHandle2, waitHandle3 };

// 在另一个线程中设置某个事件
Task.Run(() => {
    Thread.Sleep(1000);
    waitHandle2.Set();  // 触发第二个事件
});

// 等待任意一个事件触发
int signaledIndex = WaitHandle.WaitAny(handles);

Console.WriteLine($"触发的句柄索引: {signaledIndex}");  // 输出: 1 (第二个句柄)

如果你熟悉C# Task 异步编程就知道,Task类也有一个waitAny,他们两很像。

典型应用场景

  1. 多任务完成等待(任一任务完成即可继续)

  2. 多事件响应处理

  3. 超时监控

  4. 资源竞争处理

2.WaitAll

WaitHandle.WaitAll 是 .NET 中用于多线程同步的关键方法,它允许线程等待所有指定的同步对象都变为信号状态。

cs 复制代码
public static bool WaitAll(WaitHandle[] waitHandles);
public static bool WaitAll(WaitHandle[] waitHandles, int millisecondsTimeout);
public static bool WaitAll(WaitHandle[] waitHandles, TimeSpan timeout);
public static bool WaitAll(WaitHandle[] waitHandles, int millisecondsTimeout, bool exitContext);

核心功能

  1. 全等待机制:只有当所有等待句柄都收到信号时才会继续执行

  2. 混合类型支持:可以同时等待不同类型的同步对象(EventWaitHandle、Mutex、Semaphore等)

  3. 超时控制:可以设置等待的超时时间

  4. 原子性操作:确保所有句柄状态被同时检查

同样给一个例子:

cs 复制代码
// 创建三个等待句柄
var handle1 = new ManualResetEvent(false);
var handle2 = new AutoResetEvent(false);
var handle3 = new Semaphore(0, 1);

WaitHandle[] handles = { handle1, handle2, handle3 };

// 在多个线程中设置事件
Task.Run(() => {
    Thread.Sleep(500);
    handle1.Set();
});

Task.Run(() => {
    Thread.Sleep(1000);
    handle2.Set();
});

Task.Run(() => {
    Thread.Sleep(1500);
    handle3.Release();
});

// 等待所有句柄触发
bool allSignaled = WaitHandle.WaitAll(handles, 2000); // 2秒超时

Console.WriteLine($"所有句柄是否都收到信号: {allSignaled}");

重要特性

  1. 同步点:常用于需要多个条件都满足才能继续执行的场景

  2. 返回值: ◦ 成功时返回 true ◦ 超时返回 false

  3. STA线程限制:不能在单线程单元(STA)中使用(如WPF/Windows Forms的UI线程)

  4. 64句柄限制:在部分.NET版本中最多支持64个句柄

典型应用场景

  1. 多任务同步(等待所有任务完成)

  2. 复杂资源申请(需要同时获取多个资源)

  3. 分布式系统协调

  4. 并行计算屏障

3.SignalAndWait

SignalAndWait 是 WaitHandle 类中一个特殊的同步方法,它原子性地执行两个操作:先对一个等待句柄发出信号,然后立即等待另一个等待句柄。

cs 复制代码
public static bool SignalAndWait(WaitHandle toSignal, WaitHandle toWaitOn);
public static bool SignalAndWait(WaitHandle toSignal, WaitHandle toWaitOn, int millisecondsTimeout, bool exitContext);

核心功能

  1. 两步原子操作: ◦ 首先对 toSignal 调用 Set()(发出信号) ◦ 然后对 toWaitOn 调用 WaitOne()(等待)

  2. 线程优先级提升: ◦ 执行此操作时,当前线程会暂时提升优先级,减少上下文切换

  3. 超时控制: ◦ 可以设置等待的超时时间

3.1线程回合模式

cs 复制代码
// 创建两个事件对象
var wh1 = new AutoResetEvent(false);
var wh2 = new AutoResetEvent(false);

// 线程A
Task.Run(() => {
    Console.WriteLine("线程A执行工作1");
    WaitHandle.SignalAndWait(wh1, wh2);  // 通知wh1并等待wh2
    Console.WriteLine("线程A执行工作2");
});

// 线程B
Task.Run(() => {
    Console.WriteLine("线程B执行工作1");
    WaitHandle.SignalAndWait(wh2, wh1);  // 通知wh2并等待wh1
    Console.WriteLine("线程B执行工作2");
});

/* 输出顺序:
   线程A执行工作1
   线程B执行工作1
   (线程汇合)
   线程A执行工作2
   线程B执行工作2
*/

3.2 生产消费模式

cs 复制代码
var itemReady = new AutoResetEvent(false);
var spaceAvailable = new AutoResetEvent(true);

// 生产者
void Producer() {
    while(true) {
        WaitHandle.SignalAndWait(spaceAvailable, itemReady);
        // 生产物品...
    }
}

// 消费者
void Consumer() {
    while(true) {
        WaitHandle.SignalAndWait(itemReady, spaceAvailable);
        // 消费物品...
    }
}

SignalAndWait 提供了一种高效的线程协调机制,特别适合需要精确控制线程执行顺序的场景,但使用时需要特别注意其限制和潜在的死锁风险。

相关推荐
SimonKing8 分钟前
弃用MinIO,拥抱全新一代分布式文件系统RustFS
java·后端·程序员
eqwaak09 分钟前
科技信息差(9.2)
开发语言·人工智能·科技·华为·语言模型·开源
明月醉窗台13 分钟前
Qt Demo(3) 之 deepseek 帮我写的关于图像显示的小界面
开发语言·人工智能·windows·qt
银二码14 分钟前
flutter踩坑插件:Swift架构不兼容
开发语言·flutter·swift
AAA修煤气灶刘哥18 分钟前
后端人必看!配置中心这玩意儿,用 Nacos 玩明白能少熬 3 个夜
java·后端·spring cloud
一蓑烟雨,一任平生24 分钟前
h5实现内嵌微信小程序&支付宝 --截图保存海报分享功能
开发语言·前端·javascript
Ares-Wang37 分钟前
JavaScript》》JS》》ES6》》 数组 自定义属性
开发语言·javascript·es6
七夜zippoe38 分钟前
Java 技术支撑 AI 系统落地:从模型部署到安全合规的企业级解决方案(一)
java·人工智能·安全
小蒜学长39 分钟前
vue飞自在酒店管理系统(代码+数据库+LW)
java·数据库·spring boot·后端
要做朋鱼燕1 小时前
【C++】 Vector容器操作全解析
开发语言·c++·笔记·学习笔记