默认情况下,一个线程的特定指令相对于另一个线程中的指令的执行时机是不确定的,如果想要对这种不确定性进行控制,其中一种办法就是使用重置事件(虽然称为事件,但是跟C#的委托跟事件没关系)。
重置事件用于强迫代码等候另一个线程的执行,直到获得事件已发生的通知。
重置事件类型包括ManualResetEvent
、ManualResetEventSlim
以及AutoResetEvent
,但在使用过程中应该尽量用前面两种类型,而避免使用AutoResetEvent
。
ManualResetEvent
ManualResetEvent
对象具有两种状态,非信号状态及信号状态。
- 当
ManualResetEvent
对象处于非信号状态,调用WaitOne
方法,则当前线程会被阻塞,直到此ManualResetEvent
对象调用Set
方法后线程才被释放。 - 当
ManualResetEvent
对象处于信号状态,调用WaitOne
方法时,当前线程不会被阻塞,ManualResetEvent
立即释放线程并返回到非信号状态。
一、常用成员
构造函数
ManualResetEvent(Boolean)
:ManualResetEvent
的构造函数,参数为false
则创建的ManualResetEvent
对象为非信号状态,反之为信号状态。
常用方法
boolean WaitOne()
:如果当前ManualResetEvent
对象为非信号状态,阻塞当前线程,直到收到信号(调用Set()
方法);如果当前的ManualResetEvent
对象为信号状态,则不会阻塞当前线程。
- 收到信号或当前对象为信号状态时返回
true
,否则一直等待,不会返回。
bool WaitOne([int millisecondsTimeout])
:如果当前ManualResetEvent
对象为非信号状态,阻止当前线程,直到超过指定时间或收到信号(某处调用了此对象的Set()
);如果当前的ManualResetEvent
对象为信号状态,则不会阻塞当前线程。
- 收到信号返回
true
,否则返回false
。
bool Set()
:将状态设置为信号状态,允许一个或多个等待线程继续。
- 操作成功返回
true
,否则false
。
bool Reset()
:将状态设置为非信号状态。
- 如果该操作成功,返回
true
,否则false
。
Dispose()
:释放WaitHandle
类(ManualResetEvent
为其子类)的当前实例所使用的所有资源。
二、示例
ManualResetEvent manualResetEvent= new ManualResetEvent(false);Socket server = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
server.Bind(new IPEndPoint(IPAddress.Any, 9090));
server.Listen();server.BeginAccept(new AsyncCallback(asyncResult =>
{Socket s = asyncResult.AsyncState as Socket;Socket ct = s.EndAccept(asyncResult);while (true){manualResetEvent.Reset();byte[] buffer = new byte[1024];ct.BeginReceive(buffer, 0, 1024, SocketFlags.None, new AsyncCallback(asyncResult =>{Socket c = asyncResult.AsyncState as Socket;c.EndReceive(asyncResult);manualResetEvent.Set();}), ct);manualResetEvent.WaitOne();Console.WriteLine(Encoding.UTF8.GetString(buffer));}
}), server);Socket client = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
client.Connect(new IPEndPoint(IPAddress.Parse("127.0.0.1"), 9090));
for (int i = 0; i < 10; i++)
{//Thread.Sleep(20); //如果有个延时就可以看到不粘包client.Send(Encoding.UTF8.GetBytes("Hello"));
}Console.ReadLine();
三、注意
释放资源
官方文档中说明:实现 IDisposable
接口。 在使用完类型后,应直接或间接释放类型。 若要直接释放类型,请在 try
/catch
块中调用其Dispose
方法。 若要间接释放类型,请使用 using
。 因此在使用过程中,可以的话尽量考虑资源的释放。
ManualResetEventSlim
ManualResetEvent
和ManualResetEventSlim
的区别在于,前者默认使用核心同步,而后者进行了优化,除非万不得已,否则会尽量避免使用核心机制,因此ManualResetEventSlim
的性能更好。因此一般情况下应选用ManualResetEventSlim
,除非要等待多个事件或需跨越多个进程。
此外,用了一下发现,ManualResetEventSlim
用的是Wait()
而不是WaitOne()
,两者比较大的区别在于Wait()
还可以接收一个取消令牌。
AutoResetEvent
AutoResetEvent
类表示线程同步事件在一个等待线程释放后收到信号时自动重置
注意,AutoResetEvent
对象调用Set()
后会自动调用Reset()
方法,这也是类名的由来,而ManualResetEvent
其功能跟AutoResetEvent
是一样的,只是在调用Set()
方法后不会自动调用Reset()
方法。此外,AutoResetEvent
只解除一个线程的Wait()
调用所造成的阻塞,使用自动重置事件,很容易在编写生产者线程时发生失误,导致它的迭代次数多于消费者线程。
实际上,应该尽量避免使用AutoResetEvent
,而选择使用ManualResetEvent
和ManualResetEventSlim
。
- 官方示例
using System;
using System.Threading;// Visual Studio: Replace the default class in a Console project with
// the following class.
class Example
{private static AutoResetEvent event_1 = new AutoResetEvent(true);private static AutoResetEvent event_2 = new AutoResetEvent(false);static void Main(){Console.WriteLine("Press Enter to create three threads and start them.\r\n" +"The threads wait on AutoResetEvent #1, which was created\r\n" +"in the signaled state, so the first thread is released.\r\n" +"This puts AutoResetEvent #1 into the unsignaled state.");Console.ReadLine();for (int i = 1; i < 4; i++){Thread t = new Thread(ThreadProc);t.Name = "Thread_" + i;t.Start();}Thread.Sleep(250);for (int i = 0; i < 2; i++){Console.WriteLine("Press Enter to release another thread.");Console.ReadLine();event_1.Set();Thread.Sleep(250);}Console.WriteLine("\r\nAll threads are now waiting on AutoResetEvent #2.");for (int i = 0; i < 3; i++){Console.WriteLine("Press Enter to release a thread.");Console.ReadLine();event_2.Set();Thread.Sleep(250);}// Visual Studio: Uncomment the following line.//Console.Readline();}static void ThreadProc(){string name = Thread.CurrentThread.Name;Console.WriteLine("{0} waits on AutoResetEvent #1.", name);event_1.WaitOne();Console.WriteLine("{0} is released from AutoResetEvent #1.", name);Console.WriteLine("{0} waits on AutoResetEvent #2.", name);event_2.WaitOne();Console.WriteLine("{0} is released from AutoResetEvent #2.", name);Console.WriteLine("{0} ends.", name);}
}
- 实际用例
下面是下载文件的一部分代码,遍历文件列表,每一个文件开始下载后就阻塞当前线程,直到文件下载完成后,再释放线程。
private AutoResetEvent autoResetEvent = new AutoResetEvent(false);
private void OnStart()
{Task.Run(() =>{Files.ToList().ForEach(f =>{WebAccess webAccess = new WebAccess();webAccess.DownloadProgressChanged = (int progressPercent) =>{Progress = progressPercent; //设置进度泡泡};webAccess.DownLoadCompleted = () =>{autoResetEvent.Set();};webAccess.Download(f, $@".\{f.FileName}");autoResetEvent.WaitOne();});});
}