C#线程控制ManualResetEvent和AutoResetEvent

ManualResetEvent和AutoResetEvent在C#中用法比较类似,都是用来做线程控制的,这个从他们的名字

也可以看出。

先说相似点,他们都有对象方法:Set、Reset、WaitOne,用法类似,其中:

Set表示设置为有信号状态,这时调用WaitOne的线程将继续执行;

Reset表示设置为无信号状态,这时调用WaitOne的线程将阻塞;

WaitOne表示在无信号状态时阻塞当前线程,也就是说WaitOne只有在无信号状态下才会阻塞线程。

再说说不同点,

“Manual”表示手动的,“Auto”表示自动的,那么这个手动和自动的具体含义是什么呢?请看下面的例子:

 public class ResetEventTest
    {

       public void Test()
       {
           Thread t1 = new Thread(new ThreadStart(thread1));
           Thread t2 = new Thread(new ThreadStart(thread2));
           t1.Start();
           t2.Start();

           Thread.Sleep(1000); //让执行过程停止1秒

           _manualResetEvent.Set(); //设置为有信号,如果把该行代码注释掉thread1 completed!和thread2 completed!是不会执行的,为什么?

           Console.ReadLine();
       }


       private ManualResetEvent _manualResetEvent = new ManualResetEvent(false);//此处是设置初始状态false表示无信号状态,true表示有信号状态

       private void thread1()
       {
           Console.WriteLine("thread1 started!");
           _manualResetEvent.WaitOne();
           //_manualResetEvent.Reset();
           Console.WriteLine("thread1 completed!");
       }

       private void thread2()
       {
           Console.WriteLine("thread2 started!");
           _manualResetEvent.WaitOne();
           //_manualResetEvent.Reset(); //设置为无信号
           Console.WriteLine("thread2 completed!");
       }
    }
上面代码的执行结果为:

thread1 started!

thread2 started!

thread1 completed!

thread2 completed!

如果把上例中的ManualResetEvent改为AutoResetEvent可以看到thread1 completed!和thread2 completed!

其中一个是不会执行的,这是为什么呢?

因为AutoResetEvent调用Set后会在第一个调用WaitOne后自动将信号置为无信号状态导致其他调用WaitOne

的线程继续阻塞,而ManualResetEvent不会自动设置信号的状态;在上例中把_manualResetEvent.Reset()前

面的注释去掉,再运行会得到同样的效果。

你可能感兴趣的:(C#线程控制ManualResetEvent和AutoResetEvent)