AutoResetEvent 类
官方描述:通知正在等待的线程已发生事件
命名空间:System.Threading
程序集:mscorlib
继承于:System.Threading.WaitHandle
AutoResetEvent从字面理解就是自动重置事件,那么它具体做什么的呢?举个例子:大家都坐过动车,要上车之前大家都要经过检票口的一道自动检票门,插入一张车票门就打开,人走过去之后门就自动关闭,保证一张车票过一个人,那么AutoResetEvent的作用就是这道自动检票门!
简单了解了AutoResetEvent的作用后,我们来看看常用的函数:
- AutoResetEvent(bool initialState) 构造函数,initialState参数设置门的初始状态;true 初始化开着门,就是第一个通过的人不需要票,false 初始化关着门,众生平等都要票。
- WaitOne() 阻塞当前线程,直到接收到信号;官方说明有点错误,阻不阻塞线程取决于当前门的状态,如果门开着就不阻塞,反之则阻塞,如果初始化initialState设置为ture,那么第一次调用该函数是不会阻塞的,当然调用该函数的地方就是安装自动检票门的地方; WaitOne还提供间隔时间timeout的重载函数,作用就是在该间隔时间内没有插入票,那么门将自动打开。
- Set() 将事件状态设置为终止状态,允许一个或多个等待线程继续;该函数就是插入票的动作,命令开门,直到一个人通过后关闭;允许多个等待线程继续的意思就是同一个AutoResetEvent实例在多处调用WaitOne函数时,Set()函数将把这些门全部打开;该函数返回一个bool类型的值,ture表示门已经成功打开,false表示门本来就是开着的什么都没发生。
- Reset() 将事件状态设置为非终止状态,导致线程阻止;该函数是手动关闭门的命令,返回bool类型的值,ture表示门已经成功关闭,false表示门本来就是关闭的什么都没发生
示例:使用AutoResetEvent代替Thread.Sleep实现列队异步工作,来减少获取线程的获取的时间片
public class QueueWork<T> : IDisposable
{
private readonly Queue<T> _queue = new Queue<T>();
private readonly Thread _workerThread;
private readonly object _locker = new object();
private readonly AutoResetEvent _waitEvent;
private readonly Action<T> _workHandler; public QueueWork(Action<T> workHandler){
_workerThread = new Thread(Work);
_waitEvent = new AutoResetEvent(false);
_workHandler = workHandler;
} public void Add(T data){
lock(_locker){
_queue.Enqueue(data);
}
_waitEvent.Set();
} private static void Work(){
while (!_isDisposed)
{
T data;
lock(_locker){
if(_queue.Count>0){
data = _queue.Dequeue()
}
} if(data == null){
_waitEvent.WaitOne();
continue;
} try{
workHandler(data);
}catch{}
}
} private bool _isDisposed = false;
public void Dispose(){
if(!_isDisposed){
_waitEvent.Set();
_workerThread.Join();
_waitEvent.Dispose();
}
}
}