上一节主要介绍了使用锁进行同步,本节主要介绍使用信号量进行同步编程
EventWaitHandle主要用于实现信号灯机制。信号灯主要用于通知等待的线程。主要有两种实现:AutoResetEvent和ManualResetEvent。安全
AutoResetEvent从字面上理解是一个自动重置的时间。举个例子,假设有不少人等在门外,AutoResetEvent更像一个十字旋转门,每一次只容许一我的进入,进入以后门仍然是关闭状态。多线程
下面的例子演示了使用方式:线程
using System; using System.Threading; class BasicWaitHandle { static EventWaitHandle _waitHandle = new AutoResetEvent(false); static void Main() { for (int i = 0; i < 3; i++) new Thread(Waiter).Start(); for (int i = 0; i < 3; i++) { Thread.Sleep(1000); // Pause for a second... Console.WriteLine("通知下一个线程进入"); _waitHandle.Set(); // Wake up the Waiter. } Console.ReadLine(); } static void Waiter() { var threadId = Thread.CurrentThread.ManagedThreadId; Console.WriteLine("线程 {0} 正在等待", threadId); _waitHandle.WaitOne(); // 等待通知 Console.WriteLine("线程 {0} 获得通知,能够进入", threadId); } }
某些状况下,若是你连续的屡次使用Set方法通知工做线程,这个时候工做线程可能尚未准备好接收信号,这样的话后面的几回Set通知可能会没有效果。这种状况下,你须要让主线程获得工做线程接收信息的通知再开始发送信息。你可能须要经过两个信号灯实现这个功能。orm
示例代码:blog
using System; using System.Threading; class TwoWaySignaling { static EventWaitHandle _ready = new AutoResetEvent(false); static EventWaitHandle _go = new AutoResetEvent(false); static readonly object _locker = new object(); static string _message; static void Main() { new Thread(Work).Start(); _ready.WaitOne(); // 在工做线程准备接收信息以前须要一直等待 lock (_locker) _message = "床前明月光"; _go.Set(); // 通知工做线程开始工做 _ready.WaitOne(); lock (_locker) _message = "疑是地上霜"; _go.Set(); _ready.WaitOne(); lock (_locker) _message = "结束"; // 告诉工做线程退出 _go.Set(); Console.ReadLine(); } static void Work() { while (true) { _ready.Set(); // 表示当前线程已经准备接收信号 _go.WaitOne(); // 工做线程等待通知 lock (_locker) { if (_message == "结束") return; // 优雅的退出~-~ Console.WriteLine(_message); } } } }
生产消费队列是多线程编程里常见的的需求,他的主要思路是:队列
示例代码:进程
using System; using System.Threading; using System.Collections.Generic; class ProducerConsumerQueue : IDisposable { EventWaitHandle _wh = new AutoResetEvent (false); Thread _worker; readonly object _locker = new object(); Queue<string> _tasks = new Queue<string>(); public ProducerConsumerQueue() { _worker = new Thread (Work); _worker.Start(); } public void EnqueueTask (string task) { lock (_locker) _tasks.Enqueue (task); _wh.Set(); } public void Dispose() { EnqueueTask (null); // Signal the consumer to exit. _worker.Join(); // Wait for the consumer's thread to finish. _wh.Close(); // Release any OS resources. } void Work() { while (true) { string task = null; lock (_locker) if (_tasks.Count > 0) { task = _tasks.Dequeue(); if (task == null) return; } if (task != null) { Console.WriteLine ("Performing task: " + task); Thread.Sleep (1000); // simulate work... } else _wh.WaitOne(); // No more tasks - wait for a signal } } }
为了保证线程安全,咱们使用lock来保护Queue<string>集合。咱们也显示的关闭了WaitHandle。事件
在.NET 4.0中,一个新的类BlockingCollection实现了相似生产者消费者队列的功能。字符串
ManualResetEvent从字面上看是一个须要手动关闭的事件。举个例子:假设有不少人等在门外,它像是一个普通的门,门开启以后,全部等在门外的人均可以进来,当你关闭门以后,再也不容许外面的人进来。
示例代码:
using System; using System.Threading; class BasicWaitHandle { static EventWaitHandle _waitHandle = new ManualResetEvent(false); static void Main() { for (int i = 0; i < 3; i++) new Thread(Waiter).Start(); Thread.Sleep(1000); // Pause for a second... Console.WriteLine("门已打开,线程进入"); _waitHandle.Set(); // Wake up the Waiter. new Thread(Waiter).Start(); Thread.Sleep(2000); _waitHandle.Reset(); Console.WriteLine("门已关闭,线程阻塞"); new Thread(Waiter).Start(); Console.ReadLine(); } static void Waiter() { var threadId = Thread.CurrentThread.ManagedThreadId; Console.WriteLine("线程 {0} 正在等待", threadId); _waitHandle.WaitOne(); // 等待通知 Console.WriteLine("线程 {0} 获得通知,能够进入", threadId); } }
ManualResetEvent能够在当前线程唤醒全部等待的线程,这一应用很是重要。
CountdownEvent的使用和ManualEvent正好相反,是多个线程共同唤醒一个线程。
示例代码:
using System; using System.Threading; class CountDownTest { static CountdownEvent _countdown = new CountdownEvent(3); static void Main() { new Thread(SaySomething).Start("I am thread 1"); new Thread(SaySomething).Start("I am thread 2"); new Thread(SaySomething).Start("I am thread 3"); _countdown.Wait(); // 当前线程被阻塞,直到收到 _countdown发送的三次信号 Console.WriteLine("All threads have finished speaking!"); Console.ReadLine(); } static void SaySomething(object thing) { Thread.Sleep(1000); Console.WriteLine(thing); _countdown.Signal(); } }
EventWaitHandle的构造方法容许建立一个命名的EventWaitHandle,来实现跨进程的信号量操做。名字只是一个简单的字符串,只要保证不会跟其它进程的锁冲突便可。
示例代码:
EventWaitHandle wh = new EventWaitHandle(false, EventResetMode.AutoReset, "MyCompany.MyApp.SomeName");
若是两个进程运行这段代码,信号量会做用于两个进程内全部的线程。