AutoResetEvent和ManualResetEvent(多线程操作)
摘自风中灵药的博客:https://www.cnblogs.com/qingyun163/archive/2013/01/05/2846633.html#!comments
AutoResetEvent和ManualResetEvent可用于控制线程暂停或继续,拥有重要的三个方法:WaitOne、Set和Reset。
如果把每个线程比作一辆汽车的话,AutoResetEvent和ManualResetEvent就是公路上的收费站。
其中:
Reset 关闭收费站车闸禁止通行(拦截车辆才好收费啊);
WaitOne 收费员等待下一辆车辆过来(然后收费);
Set 开启收费站车闸放行(交钱了就让过去)。
AutoResetEvent和ManualResetEvent的区别
既然AutoResetEvent和ManualResetEvent都是收费站,那么它们之间有什么不同之处吗?
顾名思义,Auto即自动,Manual即手动,而Reset根据上面的比喻表示关闭车闸,也就是前者可自动关闭车闸,后者需手动关闭车闸。
自动关闭车闸:即一辆车交钱通过后,车闸会自动关闭,然后再等待下一辆车过来交费。即每辆车都要经过这么几个步骤:被阻 > 交费 > 通行 > 车闸关闭
手动关闭车闸:车闸打开后,车闸不会自动关闭,如果不手动关闭车闸(即调用ManualResetEvent.Reset()方法)的话,车辆会一辆接一辆地通过……
AutoResetEvent和ManualResetEvent的初始状态
通过设置AutoResetEvent和ManualResetEvent构造函数可初始化收费站车闸状态:
new Auto/ManualResetEvent(false):车闸默认关闭;
new Auto/ManualResetEvent(true): 车闸默认开启。
如果new Auto/ManualResetEvent(true),即车闸默认开启的话,是默认让第一辆车通过,然后会拦截下一辆车。
- static EventWaitHandle _tollStation = new AutoResetEvent(true);//车闸默认开启
- static void Main(string[] args)
- {
- new Thread(Car1).Start();
- Console.ReadKey();
- }
- static void Car1()
- {
- _tollStation.WaitOne();//因车闸默认开启,WaitOne毫无意义,不会阻止车辆前行
- Console.WriteLine("噫!车闸是开的,我过来了!");
- }
此时输出:"噫!车闸是开的,我过来了!"
但是如果改一下
- static EventWaitHandle _tollStation = new AutoResetEvent(true);//车闸默认开启
- static void Main(string[] args)
- {
- new Thread(Car1).Start();
- new Thread(Car1).Start();//执行两次输出的线程
- Console.ReadKey();
- }
- static void Car1()
- {
- _tollStation.WaitOne();//因车闸默认开启,WaitOne毫无意义,不会阻止车辆前行
- Console.WriteLine("噫!车闸是开的,我过来了!");
- }
此时输出:噫!车闸是开的,我过来了!
执行两次线程,但是只会输出一次。因为第二次被阻拦了。
如果将new AutoResetEvent(true) 改为new AutoResetEvent(flase),即车闸默认为关闭状态的话,将不会打印任何值,即车辆无法通过。
那如何才能通过呢?必须在主线程中调用Set方法,即打开车闸即可通过。
- static EventWaitHandle _tollStation = new AutoResetEvent(false);//车闸默认关闭
- static void Main(string[] args)
- {
- new Thread(Car1).Start();
- _tollStation.Set();//开启车闸
- Console.ReadKey();
- }
- static void Car1()
- {
- _tollStation.WaitOne();//等待开启车闸,即_event.Set();
- Console.WriteLine("车闸开启,我过来了!");
- }
运行将打印:
车闸开启,我过来了!
代码很明了,就不解释了,总之就是车闸默认关闭状态下,只有打开车闸(调用Set方法 ),车辆才能通行。
用代码阐释AutoResetEvent的特性
- static EventWaitHandle _tollStation = new AutoResetEvent(false);//车闸默认关闭
- static void Main(string[] args)
- {
- new Thread(Car1).Start();//车辆1
- new Thread(Car2).Start();//车辆2
- _tollStation.Set();
- Console.ReadKey();
- }
- static void Car1()
- {
- _tollStation.WaitOne();//等待开启车闸,即_tollStation.Set();
- Console.WriteLine("车辆1,顺利通过。");
- }
- static void Car2()
- {
- _tollStation.WaitOne();
- Console.WriteLine("车辆2,顺利通过。!");
- }
运行将打印:
车辆1,顺利通过。
虽然车辆1和车辆2都在运行,但只有车辆1顺利通过。
因为_tollStation.Set()仅运行了一次,即车辆1通过后车闸被立即关闭,导致车辆2未被通过。
除非,在车辆1通过后再调用一次_tollStation.Set(),即再次打开车闸,车辆2才能通过:
- static EventWaitHandle _tollStation = new AutoResetEvent(false);//车闸默认关闭
- static void Main(string[] args)
- {
- new Thread(Car1).Start();//车辆1
- new Thread(Car2).Start();//车辆2
- _tollStation.Set();//开启车闸,让车辆1通过
- Console.ReadKey();
- }
- static void Car1()
- {
- _tollStation.WaitOne();//等待开启车闸,即_tollStation.Set();
- Console.WriteLine("车辆1,顺利通过。");
- _tollStation.Set();//再开启一次车闸,让车辆2通过
- }
- static void Car2()
- {
- _tollStation.WaitOne();
- Console.WriteLine("车辆2,顺利通过。");
- }
运行将打印:
车辆1,顺利通过。
车辆2,顺利通过。
也就是每调用一次Set,仅有一个线程会继续。换言之,有多少个线程就要调用多少次Set,线程才会全部继续。
这也表明,AutoResetEvent是典型的队列操作形式。
用代码阐释ManualResetEvent的特性
在上一个代码块中,_tollStation.Set()调用了两次,两辆车才顺利通过。
那么,有没有什么办法,只调用一次_tollStation.Set()就让两辆或更多辆汽车顺利通过呢?
答案是,将AutoResetEvent改为ManualResetEvent:
- static EventWaitHandle _tollStation = new ManualResetEvent(false);//改为ManualResetEvent,车闸默认关闭
- static void Main(string[] args)
- {
- new Thread(Car1).Start();//车辆1
- new Thread(Car2).Start();//车辆2
- _tollStation.Set();//开启车闸,所有车辆都会通过
- Console.ReadKey();
- }
- static void Car1()
- {
- _tollStation.WaitOne();//等待开启车闸,即_tollStation.Set();
- Console.WriteLine("车辆1,顺利通过。");
- //_tollStation.Set();//这里不再需要了
- }
- static void Car2()
- {
- _tollStation.WaitOne();
- Console.WriteLine("车辆2,顺利通过。");
- }
运行将打印:
车辆1,顺利通过。
车辆2,顺利通过。
这很好的说明了,ManualResetEvent开启车闸后不会自动关闭这一特性。所以调用一次_tollStation.Set(),全部车辆将顺利通过。
如果在某一时刻手动关闭了车闸,则后面的车辆将无法通过。如以下代码:
- static EventWaitHandle _tollStation = new ManualResetEvent(false);//改为ManualResetEvent,车闸默认关闭
- static void Main(string[] args)
- {
- new Thread(Car1).Start();//车辆1
- new Thread(Car2).Start();//车辆2
- _tollStation.Set();//开启车闸,放行
- Timer timer = new Timer(CloseDoor, null, , );//2秒后关闭车闸
- Console.ReadKey();
- }
- static void Car1()
- {
- _tollStation.WaitOne();//等待开启车闸,即_tollStation.Set();
- Console.WriteLine("车辆1,顺利通过。");
- }
- static void Car2()
- {
- Thread.Sleep();//睡眠3秒
- _tollStation.WaitOne();//当醒来后车闸已经被关闭
- Console.WriteLine("车辆2,顺利通过。");//所以车辆2不会被通过
- }
- /// <summary>
- /// 2秒后关闭车闸
- /// </summary>
- static void CloseDoor(object o)
- {
- _tollStation.Reset();//关闭车闸
- }
运行将打印:
车辆1,顺利通过。
而车辆2将不会通过,因为当车辆2醒来时,车闸在2秒前已被关闭。
AutoResetEvent和ManualResetEvent(多线程操作)的更多相关文章
- AutoResetEvent 和 ManualResetEvent 多线程应用
AutoResetEvent 1.用于在多线程,对线程进行阻塞放行 static AutoResetEvent auth0 = new AutoResetEvent(false); static Au ...
- 多线程间通信之AutoResetEvent和ManualResetEvent的原理分析和开发示例
AutoResetEvent 允许线程通过发信号互相通信. 通常,当线程需要独占访问资源时使用该类. 线程通过调用 AutoResetEvent 上的 WaitOne 来等待信号. 如果 AutoRe ...
- C#多线程同步事件及等待句柄AutoResetEvent 和 ManualResetEvent
最近捣鼓了一下多线程的同步问题,发现其实C#关于多线程同步事件处理还是很灵活,这里主要写一下,自己测试的一些代码,涉及到了AutoResetEvent 和 ManualResetEvent,当然还有也 ...
- 多线程间通信之AutoResetEvent和ManualResetEvent的原理分析
AutoResetEvent 允许线程通过发信号互相通信. 通常,当线程需要独占访问资源时使用该类. 线程通过调用 AutoResetEvent 上的 WaitOne 来等待信号. 如果 AutoRe ...
- c# 多线程编程中AutoResetEvent和ManualResetEvent
作为等同于Java的wait,notify,notifyAll的存在,AutoResetEvent和ManualResetEvent分别实现了notify和notifyAll的功能,下面的代码简单讲解 ...
- C# 应用 - 多线程 6) 处理同步数据之手动同步 AutoResetEvent 和 ManualResetEvent
1. 类的关系 AutoResetEvent 和 ManualResetEvent 都继承自 System.Threading.EventWaitHandle 类(EventWaitHandle 继承 ...
- C#多线程操作界面控件的解决方案(转)
C#中利用委托实现多线程跨线程操作 - 张小鱼 2010-10-22 08:38 在使用VS2005的时候,如果你从非创建这个控件的线程中访问这个控件或者操作这个控件的话就会抛出这个异常.这是微软为了 ...
- 线程同步(AutoResetEvent与ManualResetEvent)
前言 在我们编写多线程程序时,会遇到这样一个问题:在一个线程处理的过程中,需要等待另一个线程处理的结果才能继续往下执行.比如:有两个线程,一个用来接收Socket数据,另一个用来处理Socket数据, ...
- WaitHandle、AutoResetEvent、ManualResetEvent
多线程中的锁系统(三)-WaitHandle.AutoResetEvent.ManualResetEvent 介绍 本章主要说下基于内核模式构造的线程同步方式,事件,信号量. 目录 一:理论 二:Wa ...
随机推荐
- libusb_transfer
http://libusb.sourceforge.net/api-1.0/structlibusb__transfer.html int libusb_control_transfer(libusb ...
- QT容器类
QList<T> 是一个指针数组,可以使用下标 append(), prepend(), insert() QList<QString> list; list<<& ...
- java面试题(目录版)
在https://www.cnblogs.com/marsitman/p/9539369.html 根据自己以往的面试经验,在该基础上做了补充和删减,均链接到相应的地址(链接失效请留言评论). 一. ...
- Python的f.seek(offset, whence)函数
file.seek()方法标准格式是:seek(offset,whence=0)offset:开始的偏移量,也就是代表需要移动偏移的字节数whence:给offset参数一个定义,表示要从哪个位置开始 ...
- windows 控制台 命令行指令换行
由于屏幕较小而指令太长,在书写的时候不好阅读. 其实cmd和Linux一样可以通过特殊符号换行的: Linux 为 \ 而cmd 为 ^ 看图 这就很美观了,拯救了我的强迫症 ^_^ ...
- php+大文件管理
用过浏览器的开发人员都对大文件上传与下载比较困扰,之前遇到了一个php文件夹上传下载的问题,无奈之下自己开发了一套文件上传控件,在这里分享一下.希望能对你有所帮助.此控件PC全平台支持包括mac,li ...
- 手机存储器在Ubuntu中的挂载位置
造冰箱的大熊猫,本文适用于Ubuntu 16.04和安卓手机@cnblogs 2018/12/6 通过USB数据线将安卓手机与Ubuntu相连,并在手机上设置“USB连接方式”为“传输文件”后,Ubu ...
- Confluence 6 分享一个文件
协同合作和编辑不仅仅是发生在页面中,很多时候你需要与你的项目小组针对文档,报告,图片,表格进行协同操作.不管是针对性的市场计划或者一个完整的项目计划,你可以在 Confluence 中让你的项目小组成 ...
- c++两数组合并算法
#include <iostream> #define MAXSIZE 100 using namespace std; int combine(int a[],int b[],int c ...
- P2015 二叉苹果树,树形dp
P2015 二叉苹果树 题目大意:有一棵二叉树性质的苹果树,每一根树枝上都有着一些苹果,现在要去掉一些树枝,只留下q根树枝,要求保留最多的苹果数(去掉树枝后不一定是二叉树) 思路:一开始就很直接的想到 ...