您的位置:首页 > 大数据 > 人工智能

线程同步 - EventWaitHandle & AutoResetEvent & ManualResetEvent

2007-02-09 16:10 579 查看
EventWaitHandle 提供了自动(EventResetMode.AutoReset)和手动(EventResetMode.ManualReset)两种方式,其主要区别是否自动将同步状态设置为非终止状态,以阻塞线程。

static void EventWaitHandleTest2(EventResetMode mode)
{
EventWaitHandle r1 = new EventWaitHandle(false, mode);

Thread t1 = new Thread(delegate()
{
while (true)
{
r1.WaitOne();

Console.WriteLine("1");
Thread.Sleep(1000);

//r1.Reset();
//r1.Set();
}
}
);

t1.Start();
r1.Set();
}

当我们使用 EventWaitHandleTest2(EventResetMode.AutoReset); 调用该方法时,我们发现循环体在执行一次后并没有继续,线程被阻塞。而 EventWaitHandleTest2(EventResetMode.ManualReset); 调用时则不会阻塞,循环体无限执行下去。由此我们可以区别这两种状态的不同,自动方式会自动将同步状态设置为非终止,而手动则不会。自动状态时,我们删除 r1.Set(); 行的注释字符 (//),将状态重置为终止状态,则循环体自动进行下一轮执行;手动方式,我们将 r1.Reset(); 行的注释行删除,那么同步状态会被重置为非终止状态状态,那么线程体自然被阻塞,不会继续循环体的执行了。

.net Framework 2.0 的 AutoResetEvent 和 ManualResetEvent 继承自 EventWaitHandle,而不是 1.x 中继承自 WaitHandle。这两个类只是重写了构造方法而已。

public AutoResetEvent(bool initialState) : base(initialState, EventResetMode.AutoReset)
{
}

public ManualResetEvent(bool initialState) : base(initialState, EventResetMode.ManualReset)
{
}

接下来,我们另写一个例子。让两个线程交替执行。

using System;
using System.Collections.Generic;
using System.Text;
using System.Diagnostics;
using System.Threading;

namespace ConsoleApplication1
{
class Program
{
static void EventWaitHandleTest(EventResetMode mode)
{
EventWaitHandle r1 = new EventWaitHandle(false, mode);
EventWaitHandle r2 = new EventWaitHandle(false, mode);

Thread t1 = new Thread(delegate()
{
while (true)
{
r1.WaitOne();

Console.WriteLine("1");
Thread.Sleep(1000);

if (mode == EventResetMode.ManualReset) r1.Reset();
r2.Set();
}
}
);

Thread t2 = new Thread(delegate()
{
while (true)
{
r2.WaitOne();
Console.WriteLine("2");

if (mode == EventResetMode.ManualReset) r2.Reset();
r1.Set();
}
}
);

t1.Start();
t2.Start();

r1.Set();
}

static void Main()
{
EventWaitHandleTest(EventResetMode.AutoReset);

Console.WriteLine("Press any key to exit...");
Console.ReadKey(true);
}
}
}

我们为什么要用2个同步对象呢?试想如果用一个的话在执行完 t1 的循环体 r1.Set(); 代码时,CPU 分配给 t1 的时间片断可能还没有超时,那么自然不会把执行权交给 t2,因为会继续执行 t1 的循环体代码,因此用 2 个则可以避免这种问题。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐