C#多线程:深入了解线程同步lock,Monitor,Mutex,同步事件和等待句柄(中)
本篇继续介绍WaitHandler类及其子类 Mutex,ManualResetEvent,AutoResetEvent的用法。.NET中线程同步的方式多的让人看了眼花缭乱,究竟该怎么去理解呢?其实,我们抛开.NET环境看线程同步,无非是执行两种操作:一是互斥/加锁,目的是保证临界区代码操作的“原子性”;另一种是信号灯操作,目的是保证多个线程按照一定顺序执行,如生产者线程要先于消费者线程执行。.NET中线程同步的类无非是对这两种方式的封装,目的归根结底都可以归结为实现互斥/ 加锁或者是信号灯这两种方式,只是它们的适用场合有所不。下面我们根据类的层次结构了解WaitHandler及其子类。
1.WaitHandler
WaitHandle是Mutex,Semaphore,EventWaitHandler,AutoResetEvent,ManualResetEvent共同的祖先,它封装Win32同步句柄内核对象,也就是说是这些内核对象的托管版本。
线程可以通过调用WaitHandler实例的方法WaitOne在单个等待句柄上阻止。此外,WaitHandler类重载了静态方法,以等待所有指定的等待句柄都已收集到信号WaitAll,或者等待某一指定的等待句柄收集到信号WaitAny。这些方法都提供了放弃等待的超时间隔、在进入等待之前退出同步上下文的机会,并允许其它线程使用同步上下文。WaitHandler是C#中的抽象类,不能实例化。
2.EventWaitHandler vs. ManualResetEvent vs. AutoResetEvent(同步事件)
我们先看看两个子类ManualResetEvent和AutoResetEvent在.NET Framework中的实现:
public class EventWaitTest { private string name; //顾客姓名 //private static AutoResetEvent eventWait = new AutoResetEvent(false); private static ManualResetEvent eventWait = new ManualResetEvent(false); private static ManualResetEvent eventOver = new ManualResetEvent(false); public EventWaitTest(string name) { this.name = name; } public static void Product() { Console.WriteLine("服务员:厨师在做菜呢,两位稍等"); Thread.Sleep(2000); Console.WriteLine("服务员:宫爆鸡丁好了"); eventWait.Set(); while (true) { if (eventOver.WaitOne(1000, false)) { Console.WriteLine("服务员:两位请买单"); eventOver.Reset(); } } } public void Consume() { while (true) { if (eventWait.WaitOne(1000, false)) { Console.WriteLine(this.name + ":开始吃宫爆鸡丁"); Thread.Sleep(2000); Console.WriteLine(this.name + ":宫爆鸡丁吃光了"); eventWait.Reset(); eventOver.Set(); break; } else { Console.WriteLine(this.name + ":等着上菜无聊先玩会手机游戏"); } } } } public class App { public static void Main(string[] args) { EventWaitTest zhangsan = new EventWaitTest("张三"); EventWaitTest lisi = new EventWaitTest("李四"); Thread t1 = new Thread(new ThreadStart(zhangsan.Consume)); Thread t2 = new Thread(new ThreadStart(lisi.Consume)); Thread t3 = new Thread(new ThreadStart(EventWaitTest.Product)); t1.Start(); t2.Start(); t3.Start(); Console.Read(); } }