一、ManualResetEvent

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

  • WaitOne:该方法用于阻塞线程,默认是无限期的阻塞,支持超时阻塞,如果超时就放弃阻塞,这样也就避免了无限期等待的尴尬;
  • Set:手动修改信号量为True,也就是恢复线程执行;
  • ReSet:重置状态;
    class Program
{
public static void Main()
{
Thread t = new Thread(Run);
t.Name = "辅助线程";
t.Start(); Console.WriteLine("当前时间:{0} {1}准备执行!", DateTime.Now.TimeOfDay, t.Name);
//手动修改信号量为True,也就是恢复一个等待线程执行。
mr.Set(); Console.ReadKey();
} //一开始设置为false才会等待收到信号才执行
static ManualResetEvent mr = new ManualResetEvent(true); static void Run()
{
//线程开始执行时待命,收到信号才动身
mr.WaitOne();
Console.WriteLine("\n当前时间:{0} {1}正式执行!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name); //我想让辅助线程暂停3秒
mr.WaitOne(3000);
Console.WriteLine("\n当前时间:{0} {1}暂停3秒,但是无效!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name); //我想让辅助线程暂停
mr.Reset();
Console.WriteLine("\n当前时间:{0} {1}还是无效!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name);
}
}

  输出如下:

  

  Reset()的意思其实是重置,重置后才又能再WaitOne();

    class Program
{
public static void Main()
{
Thread t = new Thread(Run);
t.Name = "辅助线程";
t.Start(); Console.WriteLine("当前时间:{0} {1}准备执行!", DateTime.Now.TimeOfDay, t.Name);
//手动修改信号量为True,也就是恢复一个等待线程执行。
mr.Set(); Thread.Sleep(10000);
mr.Set(); Console.ReadKey();
} //一开始设置为false才会等待收到信号才执行
static ManualResetEvent mr = new ManualResetEvent(false); static void Run()
{
//线程开始执行时待命,收到信号才动身
mr.WaitOne();
Console.WriteLine("\n当前时间:{0} {1}正式执行!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name); //重置后停止才有效
mr.Reset();
//我想让辅助线程暂停3秒
mr.WaitOne(3000);
Console.WriteLine("\n当前时间:{0} {1}暂停3秒,这回有效了!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name); //重置后停止才有效
mr.Reset();
//我想让辅助线程暂停,10后由主线程再次唤醒
mr.WaitOne();
Console.WriteLine("\n当前时间:{0} {1}暂停,但会被主线程再次唤醒,这回有效了!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name    );
}
}

  输出如下:

  

二、AutoResetEvent

  AutoResetEvent与ManualResetEvent的区别在于AutoResetEvent 的WaitOne会改变信号量的值。

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

    class Program
{
public static void Main()
{
Thread t = new Thread(Run);
t.Name = "辅助线程";
t.Start(); Console.WriteLine("当前时间:{0} {1}准备执行!", DateTime.Now.TimeOfDay, t.Name); Console.ReadKey();
} static AutoResetEvent ar = new AutoResetEvent(true); static void Run()
{
var state = ar.WaitOne(1000);
Console.WriteLine("当前的信号量状态:{0}", state); state = ar.WaitOne(1000);
Console.WriteLine("再次WaitOne后现在的状态是:{0}", state);
}
}

  输出如下:

  

  假如要实现上面ManualResetEvent同样的效果,Run方法就不用手动Reset()了:

    static void Run()
{
//线程开始执行时待命,收到信号才动身
mr.WaitOne();
Console.WriteLine("\n当前时间:{0} {1}正式执行!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name); //我想让辅助线程暂停3秒
mr.WaitOne(3000);
Console.WriteLine("\n当前时间:{0} {1}暂停3秒,这回有效了!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name); //我想让辅助线程暂停,10后由主线程再次唤醒
mr.WaitOne();
Console.WriteLine("\n当前时间:{0} {1}暂停,但会被主线程再次唤醒,这回有效了!", DateTime.Now.TimeOfDay, Thread.CurrentThread.Name);
}

  少了手动Reset()代码。

三、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.Read();
} //初始可以授予2个线程信号,因为第3个要等待前面的Release才能得到信号
static Semaphore sem = new Semaphore(2, 10); static void Run1()
{
sem.WaitOne();
Console.WriteLine("大家好,我是Run1" + DateTime.Now.TimeOfDay);
} static void Run2()
{
sem.WaitOne();
Console.WriteLine("大家好,我是Run2" + DateTime.Now.TimeOfDay); //两秒后
Thread.Sleep(2000);
sem.Release();
} static void Run3()
{
sem.WaitOne();
Console.WriteLine("大家好,我是Run3" + DateTime.Now.TimeOfDay);
}
}

  输出:

  

  在以上的方法中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);
}
}

  输出如下:

  

  •   ManualResetEvent:每次可以唤醒一个或多个线程;
  •   AutoResetEvent:每次只能唤醒一个线程;

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

  1. 转载 信号量 <第六篇>

    一.ManualResetEvent 该对象有两种信号量状态True和False.构造函数设置初始状态.简单来说, 如果构造函数由true创建,则第一次WaitOne()不会阻止线程的执行,而是等待R ...

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

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

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

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

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

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

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

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

  6. RabbitMQ学习总结 第六篇:Topic类型的exchange

    目录 RabbitMQ学习总结 第一篇:理论篇 RabbitMQ学习总结 第二篇:快速入门HelloWorld RabbitMQ学习总结 第三篇:工作队列Work Queue RabbitMQ学习总结 ...

  7. 第六篇 Replication:合并复制-发布

    本篇文章是SQL Server Replication系列的第六篇,详细内容请参考原文. 合并复制,类似于事务复制,包括一个发布服务器,一个分发服务器和一个或多个订阅服务器.每一个发布服务器上可以定义 ...

  8. 第六篇 Integration Services:初级工作流管理

    本篇文章是Integration Services系列的第六篇,详细内容请参考原文. 简介在前几篇文章中,我们关注使用增量加载方式加载数据.在本篇文章,我们将关注使用优先约束管理SSIS控制流中的工作 ...

  9. 第六篇 SQL Server安全执行上下文和代码签名

    本篇文章是SQL Server安全系列的第六篇,详细内容请参考原文. SQL Server决定主体是否有必要的执行代码权限的根本途径是其执行上下文规则.这一切都可能复杂一个主体有执行代码的权限,但是却 ...

随机推荐

  1. 【转】Linux中history历史命令使用方法详解

    原文网址:http://os.51cto.com/art/201205/335040.htm 当你在玩Linux的时候,如果你经常使用命令行来控制你的Linux系统,那么有效地使用命令历史机制将会使效 ...

  2. 《Algorithms 4th Edition》读书笔记——2.4 优先队列(priority queue)-Ⅰ

    许多应用程序都需要处理有序的元素,但不一定要求他们全部有序,或者是不一定要以此就将他们排序.很多情况下我们会手机一些元素,处理当前键值最大的元素,然后再收集更多的元素,再处理当前键值最大的元素.如此这 ...

  3. SSH方式登录github出现Permission denied (publickey)

    今天在公司上传了代码,回到家pull,结果竟然出现了“Permission denied (publickey)“这种东西.第一反应是key不对,可是上次明明用key登录过,不可能不对啊,难道是文件被 ...

  4. 新手使用ThinkPHP3.2.3的命名空间问题

    ThinkPHP3.2.3的命名空间问题 命名空间的出现是为了避免命名冲突. 我们在TP3.2.3的Collection和Model的创建过程中经常会遇到这样的两行代码: 这是在控制器中的写法.其中n ...

  5. java与.net比较学习系列(5) 流程控制语句

    java中流程控制语句主要分为以下几类,第一,条件语句,主要包括if语句和switch语句.第二,循环语句,主要包括while循环语句,for循环语句.第三,跳转语句,主要包括三种,break跳出语句 ...

  6. 【Android】实现动态显示隐藏密码输入框的内容

    在设置输入密码框时,有些时候需要按钮控制输入的是“明文”或者“暗文”. 这里提供一种Android实现动态显示隐藏密码输入框的内容的方法: 主要是通过设置EditText的setTransformat ...

  7. 大到可以小说的Y组合子(三)

    答:关于Fix的问题你fix了吗? 问:慢着,让我想想,上次留下个什么问题来着?是说我们有了一个求不动点的函数Fix,但Fix却是显式递归的,是吧? 答:有劳你还记的这个问题. 问:Fix的参与背离了 ...

  8. Dreamwaver 使用root用户连接不上远程服务器

    我用dreamweaver连接远程服务,开始用的是root用户登录的,但是连接不上.网上查了一下,解决教程非常复杂,我就不列出来了. 后来我想了一下,之前我有连接过.我感觉可能是用户的问题,于是我在远 ...

  9. django中的Model模型一:

    在django的框架设计中采用了mtv模型,即Model,template,viewer Model相对于传统的三层或者mvc框架来说就相当对数据处理层,它主要负责与数据的交互,在使用django框架 ...

  10. jquery动态添加DOM节点

    1.append()方法:向每个匹配的元素内部添加元素 appendTo()方法:将所有匹配的元素追加的指定的元素中 <html> <head> <meta http-e ...