一、ManualResetEvent

该对象有两种信号量状态True和False。构造函数设置初始状态。简单来说,

  • 如果构造函数由true创建,则第一次WaitOne()不会阻止线程的执行,而是等待Reset后的第二次WaitOne()才阻止线程执行。
  • 如果构造函数有false创建,则WaitOne()必须等待Set()才能往下执行。

  一句话总结就是:是否忽略第一次阻塞。

  方法如下:

  • WaitOne:该方法用于阻塞线程,默认是无限期的阻塞,支持设置等待时间,如果超时就放弃阻塞,不等了,继续往下执行;
  • Set:手动修改信号量为True,也就是恢复线程执行;
  • ReSet:重置状态;
    class Program
{
//一开始设置为false才会等待收到信号才执行
static ManualResetEvent mr = new ManualResetEvent(false);
public static void Main()
{
Thread t = new Thread(Run);
//启动辅助线程
t.Start();
//等待辅助线程执行完毕之后,主线程才继续执行
Console.WriteLine("主线程一边做自己的事,一边等辅助线程执行!" + DateTime.Now.ToString("mm:ss"));
mr.WaitOne();
Console.WriteLine("收到信号,主线程继续执行" + DateTime.Now.ToString("mm:ss"));
Console.ReadKey();
} static void Run()
{
//模拟长时间任务
Thread.Sleep(3000);
Console.WriteLine("辅助线程长时间任务完成!" + DateTime.Now.ToString("mm:ss"));
mr.Set();
}
}

  输出结果如下:

  

  在思维上,这个东西可以有两种用法,一种是让主线程等待辅助线程,一种是辅助线程等待主线程。

  但无论怎么用,都是让一个线程等待或唤醒另外一个线程。

  Reset

    class Program
{
//一开始设置为false,当遇到WaitOne()时,需要Set()才能继续执行
static ManualResetEvent mr = new ManualResetEvent(false); public static void Main()
{
Thread t = new Thread(Run);
t.Start();
mr.WaitOne();
Console.WriteLine("第一次等待完成!" + DateTime.Now.ToString("mm:ss"));
mr.Reset(); //重置后,又能WaitOne()啦
mr.WaitOne(3000);
Console.WriteLine("第二次等待完成!" + DateTime.Now.ToString("mm:ss"));
Console.ReadKey();
} static void Run()
{
mr.Set();
Thread.Sleep(2000);
mr.Set();
}
}

  输出如下:

  

  如果以上代码不使用Reset,则直接输出第二次等待完成,而不会等待2秒。

二、AutoResetEvent

  AutoResetEvent与ManualResetEvent的区别在于AutoResetEvent 的WaitOne会改变信号量的值为false,让其等待阻塞。

  比如说初始信号量为True,如果WaitOne超时信号量将自动变为False,而ManualResetEvent则不会。

  第二个区别:

  • ManualResetEvent:每次可以唤醒一个或多个线程;
  • AutoResetEvent:每次只能唤醒一个线程;
    class Program
{
static AutoResetEvent ar = new AutoResetEvent(true);
public static void Main()
{
Thread t = new Thread(Run);
t.Start(); bool state = ar.WaitOne(1000);
Console.WriteLine("当前的信号量状态:{0}", state); state = ar.WaitOne(1000);
Console.WriteLine("再次WaitOne后现在的状态是:{0}", state); state = ar.WaitOne(1000);
Console.WriteLine("再次WaitOne后现在的状态是:{0}", state); Console.ReadKey();
} static void Run()
{
Console.WriteLine("当前时间" + DateTime.Now.ToString("mm:ss"));
}
}

  输出如下:

  

  假如要用ManualResetEvent实现上面同样的效果,Run方法就不用手动Reset()了,AutoResetEvent保证后续每个WaitOne()都有效:

    static void Run()
{
//线程开始执行时待命,收到信号才动身
mr.WaitOne();//我想让辅助线程暂停3秒
mr.WaitOne(3000);//我想让辅助线程暂停,10后由主线程再次唤醒
mr.WaitOne();
}

  少了手动Reset()代码。

  2014-11-13

  Workflow 4.5用的就是这个东西,因为对信号量这个东西不熟,可算吃了大亏。

三、Semaphore

  用于控制线程的访问数量,默认的构造函数为initialCount和maximumCount,表示默认设置的信号量个数和最大信号量个数。当你WaitOne的时候,信号量自减,当Release的时候,信号量自增,然而当信号量为0的时候,后续的线程就不能拿到WaitOne了,所以必须等待先前的线程通过Release来释放。

    class Program
{
static void Main(string[] args)
{
Thread t1 = new Thread(Run1);
t1.Start();
Thread t2 = new Thread(Run2);
t2.Start();
Thread t3 = new Thread(Run3);
t3.Start();
Console.ReadKey();
} //初始可以授予2个线程信号,因为第3个要等待前面的Release才能得到信号
static Semaphore sem = new Semaphore(2, 10); static void Run1()
{
sem.WaitOne();
Console.WriteLine("大家好,我是Run1;" + DateTime.Now.ToString("mm:ss")); //两秒后
Thread.Sleep(2000);
sem.Release();
} static void Run2()
{
sem.WaitOne();
Console.WriteLine("大家好,我是Run2;" + DateTime.Now.ToString("mm:ss")); //两秒后
Thread.Sleep(2000);
sem.Release();
} static void Run3()
{
sem.WaitOne();
Console.WriteLine("大家好,我是Run3;" + DateTime.Now.ToString("mm:ss")); //两秒后
Thread.Sleep(2000);
sem.Release();
}
}

  输出:

  

  在以上的方法中Release()方法相当于自增一个信号量,Release(5)自增5个信号量。但是,Release()到构造函数的第二个参数maximumCount的值就不能再自增了。

  命名Semaphore可用于进程级交互。

    class Program
{
static void Main(string[] args)
{ Thread t1 = new Thread(Run1);
t1.Start(); Thread t2 = new Thread(Run2);
t2.Start(); Console.Read();
} //初始可以授予2个线程信号,因为第3个要等待前面的Release才能得到信号
static Semaphore sem = new Semaphore(3, 10, "命名Semaphore"); static void Run1()
{
sem.WaitOne(); Console.WriteLine("进程:" +Process.GetCurrentProcess().Id + " 我是Run1" + DateTime.Now.TimeOfDay);
} static void Run2()
{
sem.WaitOne(); Console.WriteLine("进程:" + Process.GetCurrentProcess().Id + " 我是Run2" + DateTime.Now.TimeOfDay);
}
}

  输出如下:

  

  这个东西是跨进程的,如何测试,直接运行两次bin目录的exe文件,就能发现最多只能输出3个。

四、综合示例(线程、事件、信号量)

  要搞清楚线程、信号量、事件这三者的关系。实际上3个东西并无具体联系,各自有各自的作用,但是配合起来使用,威力无穷。

  下面用一个例子,结合事件、信号量、线程来实现如下功能:

  1. 主线程启动辅助线程执行一个长时间任务;
  2. 辅助线程完成时,触发完成事件(),调用委托,让主线程继续执行;
namespace ConsoleApplication3
{
class Program
{
static AutoResetEvent ar = new AutoResetEvent(true);
static void MyEventHandler(object sender, EventArgs e)
{
ar.Set();
} static void Main(string[] args)
{
LongTimeWork LTW = new LongTimeWork();
LTW.Completed += MyEventHandler;
Thread t = new Thread(LTW.MyLongTimeWork);
t.Start();
//继续忙我的
Thread.Sleep(2000);
//等待辅助线程完成
ar.WaitOne();
Console.WriteLine("主线程完成!");
Console.ReadKey();
}
} public class LongTimeWork
{
//定义一个事件
public event EventHandler Completed; public void MyLongTimeWork()
{
Thread.Sleep(1000);
Console.WriteLine("辅助线程长时间任务完成!");
//当辅助线程完成时,触发已完成事件
if (Completed != null)
{
Completed(this, new EventArgs());
}
}
}
}

  输出如下:

  

  以上虽然短短几十行代码,但是我却开发了两年多.Net之后才能够领悟。其主要作用是什么,以上达到了线程控制的目的,当我们开发一个核心模块时(LongTimeWork),仅仅暴露出一个事件(Completed),调用的人配合上信号量(AutoResetEvent),就能够随意调用你的核心模块。这也是WF4的调用方式。

转载 信号量 <第六篇>的更多相关文章

  1. 信号量 <第六篇>

    一.ManualResetEvent 该对象有两种信号量状态True和False.构造函数设置初始状态. WaitOne:该方法用于阻塞线程,默认是无限期的阻塞,支持超时阻塞,如果超时就放弃阻塞,这样 ...

  2. “MVC+Nhibernate+Jquery-EasyUI” 信息发布系统 第六篇(图片新闻的添加以及带分页的静态页的生成)

    “MVC+Nhibernate+Jquery-EasyUI” 信息发布系统 第六篇(图片新闻的添加以及带分页的静态页的生成) 一.这篇文章主要是要实现:图片新闻的添加,无刷新图片的上传,以及添加新闻静 ...

  3. EnjoyingSoft之Mule ESB开发教程第六篇:Data Transform - 数据转换

    目录 1. 数据转换概念 2. 数据智能感知 - DataSense 3. 简单数据转换组件 3.1 Object to JSON 3.2 JSON to XML 3.3 JSON to Object ...

  4. 小白进阶之Scrapy第六篇Scrapy-Redis详解(转)

    Scrapy-Redis 详解 通常我们在一个站站点进行采集的时候,如果是小站的话 我们使用scrapy本身就可以满足. 但是如果在面对一些比较大型的站点的时候,单个scrapy就显得力不从心了. 要 ...

  5. spring cloud系列教程第六篇-Eureka集群版

    spring cloud系列教程第六篇-Eureka集群版 本文主要内容: 本文来源:本文由凯哥Java(kaigejava)发布在博客园博客的.转载请注明 1:Eureka执行步骤理解 2:集群原理 ...

  6. 解剖SQLSERVER 第十六篇 OrcaMDF RawDatabase --MDF文件的瑞士军刀(译)

    解剖SQLSERVER 第十六篇 OrcaMDF RawDatabase --MDF文件的瑞士军刀(译) http://improve.dk/orcamdf-rawdatabase-a-swiss-a ...

  7. 解剖SQLSERVER 第六篇 对OrcaMDF的系统测试里避免regressions(译)

    解剖SQLSERVER 第六篇  对OrcaMDF的系统测试里避免regressions (译) http://improve.dk/avoiding-regressions-in-orcamdf-b ...

  8. Python之路【第十六篇】:Django【基础篇】

    Python之路[第十六篇]:Django[基础篇]   Python的WEB框架有Django.Tornado.Flask 等多种,Django相较与其他WEB框架其优势为:大而全,框架本身集成了O ...

  9. 第六篇 :微信公众平台开发实战Java版之如何自定义微信公众号菜单

    我们来了解一下 自定义菜单创建接口: http请求方式:POST(请使用https协议) https://api.weixin.qq.com/cgi-bin/menu/create?access_to ...

随机推荐

  1. 学会4种备份MySQL数据库(基本备份方面没问题了)

    前言 我们试着想一想, 在生产环境中什么最重要?如果我们服务器的硬件坏了可以维修或者换新, 软件问题可以修复或重新安装, 但是如果数据没了呢?这可能是最恐怖的事情了吧, 我感觉在生产环境中应该没有什么 ...

  2. MDK5 新建工程提示报错:Load PDSC Dubug Description faild

    File : D:\Keil_v5\ARM\PACK\Keil\STM32F0xx_DFP\2.0.0\Keil.STM32F0xx_DFP.pdsc Sequence : CheckID Conte ...

  3. 虚拟机安装ubuntu18.04及其srs服务器的搭建

    第一次写博客,有些地方可能不太完善. 1.安装VMware,我用的是VMware12. 2.下载Ubuntu镜像(自Ubuntu 17.10开始桌面版本不再提供32位安装镜像,Ubuntu Serve ...

  4. java 约束配置文件和本地约束

    一.寻找spring配置文件约束头(也可直接复制已有的) 1.在本地文件夹解压spring核心包(dist) 例:核心包的约束位置(D:\JavaSources\spring-framework-4. ...

  5. python自动化开发-5a

    python的常用模块 在计算机程序的开发过程中,随着程序代码越写越多,在一个文件里代码就会越来越长,越来越不容易维护.为了编写可维护的代码,我们把很多函数分组,分别放到不同的文件里,这样,每个文件包 ...

  6. selenium win7+selenium2.0+python环境搭建

    win7+selenium2.0+python环境搭建 by:授客 QQ:1033553122 步骤1:下载python 担心最新版的支持不太好,这里我下载的是python 2.7(selenium之 ...

  7. Python-初识模块

    #系统自带的模块 import sys print(sys.path)#打印环境变量 print(sys.argv)#打印绝对路径 import os #cmd_res = os.system(&qu ...

  8. Asp.net Mvc、webApi配置允许跨域

    Web.config 下<system.webServer> 节点下配置 <httpProtocol> <customHeaders> <add name=& ...

  9. 微信小程序搭建和开发相关指引

    几点: 1.环境搭建 2.开发和调试 3.发布 原文链接: http://www.lookdaima.com/WebForms/WebPages/Blanks/Pm/Docs/DocItemDetai ...

  10. nginx基础知识总结

    1.nginx的工作模式 master/worker工作模式: 一个master进程: 负载加载和分析配置文件.管理worker进程.平滑重启升级等. 一个或多个worker进程 处理并响应用户请求 ...