上一节主要介绍了使用锁进行同步,本节主要介绍使用信号量进行同步

使用EventWaitHandle信号量进行同步

EventWaitHandle主要用于实现信号灯机制。信号灯主要用于通知等待的线程。主要有两种实现:AutoResetEvent和ManualResetEvent。

AutoResetEvent

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通知可能会没有效果。这种情况下,你需要让主线程得到工作线程接收信息的通知再开始发送信息。你可能需要通过两个信号灯实现这个功能。

示例代码:

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);
}
}
}
}

生产消费队列

生产消费队列是多线程编程里常见的的需求,他的主要思路是:

  1. 一个队列用来存放工作线程需要用到的数据
  2. 当新的任务加入队列的时候,调用线程不需要等待工作结束
  3. 1个或多个工作线程在后台获取队列中数据信息

示例代码:

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

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

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,来实现跨进程的信号量操作。名字只是一个简单的字符串,只要保证不会跟其它进程的锁冲突即可。

示例代码:

EventWaitHandle wh = new EventWaitHandle(false, EventResetMode.AutoReset, "MyCompany.MyApp.SomeName");

  

如果两个进程运行这段代码,信号量会作用于两个进程内所有的线程。

细说.NET中的多线程 (五 使用信号量进行同步)的更多相关文章

  1. 细说.NET中的多线程 (四 使用锁进行同步)

    通过锁来实现同步 排它锁主要用来保证,在一段时间内,只有一个线程可以访问某一段代码.两种主要类型的排它锁是lock和Mutex.Lock和Mutex相比构造起来更方便,运行的也更快.但是Mutex可以 ...

  2. 细说.NET 中的多线程 (一 概念)

    为什么使用多线程 使用户界面能够随时相应用户输入 当某个应用程序在进行大量运算时候,为了保证应用程序能够随时相应客户的输入,这个时候我们往往需要让大量运算和相应用户输入这两个行为在不同的线程中进行. ...

  3. 细说.NET中的多线程 (二 线程池)

    上一章我们了解到,由于线程的创建,销毁都是需要耗费大量资源和时间的,开发者应该非常节约的使用线程资源.最好的办法是使用线程池,线程池能够避免当前进行中大量的线程导致操作系统不停的进行线程切换,当线程数 ...

  4. 细说.NET中的多线程 (六 使用MemoryBarrier,Volatile进行同步)

    上一节介绍了使用信号量进行同步,本节主要介绍一些非阻塞同步的方法.本节主要介绍MemoryBarrier,volatile,Interlocked. MemoryBarriers 本文简单的介绍一下这 ...

  5. 细说.NET中的多线程 (三 使用Task)

    上一节我们介绍了线程池相关的概念以及用法.我们可以发现ThreadPool. QueueUserWorkItem是一种起了线程之后就不管了的做法.但是实际应用过程,我们往往会有更多的需求,比如如果更简 ...

  6. NET 中的多线程

    NET 中的多线程 为什么使用多线程 使用户界面能够随时相应用户输入 当某个应用程序在进行大量运算时候,为了保证应用程序能够随时相应客户的输入,这个时候我们往往需要让大量运算和相应用户输入这两个行为在 ...

  7. C#中的多线程 - 同步基础

    原文:http://www.albahari.com/threading/part2.aspx 文章来源:http://blog.gkarch.com/threading/part2.html 1同步 ...

  8. 多线程(五) java的线程锁

    在多线程中,每个线程的执行顺序,是无法预测不可控制的,那么在对数据进行读写的时候便存在由于读写顺序多乱而造成数据混乱错误的可能性.那么如何控制,每个线程对于数据的读写顺序呢?这里就涉及到线程锁. 什么 ...

  9. Java中的 多线程编程

    Java 中的多线程编程 一.多线程的优缺点 多线程的优点: 1)资源利用率更好2)程序设计在某些情况下更简单3)程序响应更快 多线程的代价: 1)设计更复杂虽然有一些多线程应用程序比单线程的应用程序 ...

随机推荐

  1. margin:0 auto 与 text-align:center 的区别

    基本概念: 1.text-align: 属性规定元素中的文本的水平对齐方式;   该属性通过指定行框与哪个点对齐,从而设置块级元素内文本的水平对齐方式;  一般情况下设置文本对齐方式的时使用此属性.支 ...

  2. 深入研究C语言 第一篇(续)

    没有读过第一篇的读者,可以点击这里,阅读深入研究C语言的第一篇. 问题一:如何打印变量的地址? 我们用取地址符&,可以取到变量的偏移地址,用DS可以取到变量的段地址. 1.全局变量: 我们看到 ...

  3. 【Java】JDBC连接数据库

    JDBC介绍 JDBC(Java Data Base Connectivity,java数据库连接)是一种用于执行SQL语句的Java API,可以为多种关系数据库提供统一访问,它由一组用Java语言 ...

  4. C#开发Android环境搭建

    目前破解比较稳定的版本(我亲自尝试过的)是4.2. wuleba上的4.6,4.8,4.10 破解均会出现各种问题. 1 当前电脑账户最好是使用英文账号,而不要使用汉字,否则路径会出现乱码问题. 2 ...

  5. Java编程中“为了性能”尽量要做到的一些地方

    最近的机器内存又爆满了,除了新增机器内存外,还应该好好review一下我们的代码,有很多代码编写过于随意化,这些不好的习惯或对程序语言的不了解是应该好好打压打压了. 下面是参考网络资源总结的一些在Ja ...

  6. 除法取模练习(51nod 1119 & 1013 )

    题目:1119 机器人走方格 V2 思路:求C(m+n-2,n-1) % 10^9 +7       (2<=m,n<= 1000000) 在求组合数时,一般都通过双重for循环c[i][ ...

  7. 解决Android5.0以后DatePicker选择时间无效的bug。

    一.在布局中加上这句话. 加上了这句话后,就相当于强制用5.0以前的外观,所以外观会有所变化: 5.0以上没有这句话的外观: 加上之后的外观: 二.可以用DatePickerDialog代替

  8. android开发中在界面上实现曲线图的几个开源项目

    转自:https://wapiknow.baidu.com/question/1959128379041474620?qq-pf-to=pcqq.c2c 几个相关开源项目: 1.  MPAndroid ...

  9. node开发

    1. 国内使用npm安装某些插件的时候,偶尔会有网络问题,可以使用cnpm:(后续所有使用 npm 无法正常安装的,都改成 cnpm 试试) a. 首先使用 npm 安装 cnpm:npm insta ...

  10. Android密码约束规则例子一

    Android常用的一个密码规则 (一)密码必须是8至16位:(二)密码必须包含英文字母和数字:(三)密码不能包含4位连续相同的字符,如0000或AAAA:(四)密码不能包含4位连续递增或连续递减的数 ...