C#线程间互相通信主要用到两个类:AutoResetEvent和ManualResetEvent。

一、AutoResetEvent

AutoResetEvent 允许线程通过发信号互相通信,线程通过调用 AutoResetEvent 上的 WaitOne 来等待信号。 如果 AutoResetEvent 为非终止状态,则线程会被阻止,并等待当前控制资源的线程通过调用 Set 来通知资源可用。

下面我用吃快餐的例子来说明这个问题,吃快餐的时候都会排队付款,收银员发送收款通知,客户依次付钱,代码如下:

  class Program
{
//若要将初始状态设置为终止,则为 true;若要将初始状态设置为非终止,则为 false
static AutoResetEvent autoResetEvent = new AutoResetEvent(false); static void Main(string[] args)
{
Thread t1 = new Thread(() =>
{
Console.WriteLine("客户1在排队等待付钱..."); //客户1调用AutoResetEvent上的WaitOne来等待付钱通知
autoResetEvent.WaitOne();
Console.WriteLine("通知来了,客户1付钱");
});
t1.IsBackground = true;
t1.Start(); Pay();//发送通知
Console.ReadKey();
} static void Pay()
{
string tip = Console.ReadLine();
if (tip == "next")
{
autoResetEvent.Set();//收银员发送付钱通知,通过调用Set来通知客户付钱
}
}
}

运行在屏幕中打印:

客户1在排队等待付钱...

等收银员说“next”的时候,向客户1发送付钱通知(autoResetEvent.Set()),屏幕打印:

客户1在排队等待付钱...
next
通知来了,客户1付钱!

AutoResetEvent类一次只能通知一个等待的线程,且通知一次过后会立即将AutoResetEvent对象的状态置为false,也就是如果有两个客户都在等待收银员通知,AutoResetEvent对象的set方法只能通知到第一个客户,代码和效果如下:

  class Program
{
//若要将初始状态设置为终止,则为 true;若要将初始状态设置为非终止,则为 false。
static AutoResetEvent autoResetEvent = new AutoResetEvent(false); static void Main(string[] args)
{
Thread t1 = new Thread(() =>
{
Console.WriteLine("客户1在排队等待付钱..."); //客户1调用AutoResetEvent上的WaitOne来等待付钱通知
autoResetEvent.WaitOne();
Console.WriteLine("通知来了,客户1付钱");
});
t1.IsBackground = true;
t1.Start(); Thread t2 = new Thread(() =>
{
Console.WriteLine("客户2在排队等待付钱..."); //客户2调用AutoResetEvent上的WaitOne来等待付钱通知
autoResetEvent.WaitOne();
Console.WriteLine("通知来了,客户2付钱!");
});
t2.IsBackground = true;
t2.Start(); Pay();//发送通知 Console.ReadKey();
} static void Pay()
{
string tip = Console.ReadLine();
if (tip == "next")
{
autoResetEvent.Set();//收银员发送付钱通知,通过调用Set来通知客户付钱
}
}
}

运行后屏幕打印:

客户1在排队等待付钱...
客户1在排队等待付钱...
next
通知来了,客户1付钱! 

这就说明在通知完客户1后,autoResetEvent 的状态又被置为了false,这时如果要通知到客户2,就需要在通知完客户1后,再执行一次通知,在线程1中加上一行代码,如下:

  Thread t1 = new Thread(() =>
{
Console.WriteLine("客户1在排队等待付钱..."); //客户1调用AutoResetEvent上的WaitOne来等待付钱通知
autoResetEvent.WaitOne();
Console.WriteLine("通知来了,客户1付钱"); autoResetEvent.Set();//让其再通知下个客户
});

运行后屏幕打印:

客户1在排队等待付钱...
客户1在排队等待付钱...
next
通知来了,客户1付钱!
通知来了,客户2付钱! 

这也就说明每调用一次Set,只有一个线程会解除等待,换句话说,有多少个线程就要调用多少次Set,线程才会全部继续。

、ManualResetEvent

在AutoResetEvent中,如果要通知两个线程,则Set方法要被执行两次,也以快餐店的例子做了举例,但如果有一天客户1中彩票了,要请部门的10个同事吃饭,也就是说只要Set一次,所有在等待的线程都会解除等待,相当于收银员只收一次钱,10个人都可以通过收银去吃饭,这时我们就要用到ManualResetEvent类,它的用法和AutoResetEvent基本一样,区别就在于它是一量Set方法发出通知后,要再次阻塞的话就需要手动去修改,也就是调用Reset方法,代码如下:

  class Program
{
//若要将初始状态设置为终止,则为 true;若要将初始状态设置为非终止,则为 false。
static ManualResetEvent manualResetEvent = new ManualResetEvent(false); static void Main(string[] args)
{
Thread t1 = new Thread(() =>
{
Console.WriteLine("客户1在排队等待付钱..."); //客户1调用manualResetEvent上的WaitOne来等待付钱通知
manualResetEvent.WaitOne();
Console.WriteLine("已经有人请客,客户1不用付钱");
});
t1.IsBackground = true;
t1.Start(); Thread t2 = new Thread(() =>
{
Console.WriteLine("客户2在排队等待付钱..."); //客户2调用manualResetEvent上的WaitOne来等待付钱通知
manualResetEvent.WaitOne();
Console.WriteLine("已经有人请客,客户2不用付钱!");
});
t2.IsBackground = true;
t2.Start(); Pay();//发送通知 Console.ReadKey();
} static void Pay()
{
string tip = Console.ReadLine();
if (tip == "next")
{
manualResetEvent.Set();//收银员发送付钱通知,通过调用Set来通知客户付钱
}
}
}

运行后屏幕打印:

客户1在排队等待付钱...
客户1在排队等待付钱...
next
已经有人请客,客户1不用付钱!
已经有人请客,客户2不用付钱! 如果收银员在发送通知后5秒就下班了,就不能再收钱了,这时就要把通知重置掉,让没接到通知的客户继续处于继续等待,代码如下:
  class Program
{
//若要将初始状态设置为终止,则为 true;若要将初始状态设置为非终止,则为 false。
static ManualResetEvent manualResetEvent = new ManualResetEvent(false); static void Main(string[] args)
{
Thread t1 = new Thread(() =>
{
Console.WriteLine("客户1在排队等待付钱..."); //客户1调用manualResetEvent上的WaitOne来等待付钱通知
manualResetEvent.WaitOne();
Console.WriteLine("已经有人请客,客户1不用付钱");
});
t1.IsBackground = true;
t1.Start(); Thread t2 = new Thread(() =>
{
Console.WriteLine("客户2在排队等待付钱..."); Thread.Sleep();//客户2发呆了8秒,这时收银员已经下班,要继续等待
//客户2调用manualResetEvent上的WaitOne来等待付钱通知
manualResetEvent.WaitOne();
Console.WriteLine("已经有人请客,客户2不用付钱!");
});
t2.IsBackground = true;
t2.Start(); Pay();//发送通知 Console.ReadKey();
} static void Pay()
{
string tip = Console.ReadLine();
if (tip == "next")
{
manualResetEvent.Set();//收银员发送付钱通知,通过调用Set来通知客户付钱 Timer timer = new Timer(StopPay, null, , );//5秒钟后收银员下班了,线程要重新等待了
}
} static void StopPay(object s)
{
manualResetEvent.Reset();
Console.WriteLine("收银员下班, 后面的客户要继续等待");
}
}
运行后屏幕打印:
客户1在排队等待付钱...
客户1在排队等待付钱...
next
已经有人请客,客户1不用付钱!
收银员下班,后面的客户要继续等待
总结
AutoResetEvent和ManualResetEvent的主要区别就在于:AutoResetEvent一次只能通知一个等待线程,通知后自动关闭; 而ManualResetEvent一次可通知很多个等待的线程,但要关闭需要调用Reset方法手动关闭。

C# 线程间互相通信 AutoResetEvent和ManualResetEvent的更多相关文章

  1. C# 线程间互相通信

    C#线程间互相通信主要用到两个类:AutoResetEvent和ManualResetEvent. 一.AutoResetEvent AutoResetEvent 允许线程通过发信号互相通信,线程通过 ...

  2. iOS开发多线程篇—线程间的通信

    iOS开发多线程篇—线程间的通信 一.简单说明 线程间通信:在1个进程中,线程往往不是孤立存在的,多个线程之间需要经常进行通信 线程间通信的体现 1个线程传递数据给另1个线程 在1个线程中执行完特定任 ...

  3. Java核心知识点学习----多线程并发之线程间的通信,notify,wait

    1.需求: 子线程循环10次,主线程循环100次,这样间隔循环50次. 2.实现: package com.amos.concurrent; /** * @ClassName: ThreadSynch ...

  4. java多线程详解(6)-线程间的通信wait及notify方法

    Java多线程间的通信 本文提纲 一. 线程的几种状态 二. 线程间的相互作用 三.实例代码分析 一. 线程的几种状态 线程有四种状态,任何一个线程肯定处于这四种状态中的一种:(1). 产生(New) ...

  5. iOS边练边学--多线程NSOperation介绍,子类实现多线程的介绍(任务和队列),队列的取消、暂停(挂起)和恢复,操作依赖与线程间的通信

    一.NSOperation NSOperation和NSOperationQueue实现多线程的具体步骤 先将需要执行的操作封装到一个NSOperation对象中 然后将NSOperation对象添加 ...

  6. 新建线程与UI线程间的通信

    现在用一个实例来演示一下自己的新建线程与UI线程间的通信. UI界面包含3个控件: 一个输入框,用来输入数字: 一个显示框,用来显示从2开始,到输入数字之间的所有质数: 一个按钮,点击后获取输入框输入 ...

  7. QThread与其他线程间相互通信

    转载请注明链接与作者huihui1988 QThread的用法其实比较简单,只需要派生一个QThread的子类,实现其中的run虚函数就大功告成, 用的时候创建该类的实例,调用它的start方法即可. ...

  8. Handler不同线程间的通信

    转http://www.iteye.com/problems/69457 Activity启动后点击一个界面按钮后会开启一个服务(暂定为padService),在padService中会启动一个线程( ...

  9. Java多线程中线程间的通信

    一.使用while方式来实现线程之间的通信 package com.ietree.multithread.sync; import java.util.ArrayList; import java.u ...

随机推荐

  1. 别人的Linux私房菜(5)首次CentOS7与帮助等

    ctrl alt F1-F6切换终端tty1-6,其中,F1的终端带有用户界面. 在终端登录后,输入startx启动个人图形界面.(启动有一些条件限制,如没有其他的X Window启用,已经安装,并具 ...

  2. 学习newton raphson and back eluer

    % % time step https://ww2.mathworks.cn/matlabcentral/answers/184200-newton-raphson-loop-for-backward ...

  3. python基本数据类型之字符串(五)

    python基本数据类型之字符串(五) 遍历与查找 python中的字符串属于可迭代对象,通过一些方法可以遍历字符串中的每一个字符.而查找的方法主要有两个:find与index. 1.字符串的遍历 字 ...

  4. 安装VS2017后打开项目提示 asp.net 4.0尚未web服务器注册

    Visual Studio 2017 出来了,手痒安装完成后打开原来的项目缺提示,asp.net 4.0尚未web服务器注册.郁闷了… 按照提示的方法,如何:将 ASP.NET Web 应用程序升级到 ...

  5. 在 Linux 上如何挂载 qcow2 磁盘镜像

    1.下载qemu-nbd工具 sudo apt-get install qemu-utils 或者 sudo yum install qemu-img 2.加载nbd模块,然后挂载 sudo modp ...

  6. 标签页(tab)切换的原生js,jquery和bootstrap实现

    概述 这是我在学习课程Tab选项卡切换效果时做的总结和练手. 原课程中只有原生js实现,jquery和bootstrap实现是我自己补上的. 本节内容 标签页(tab)切换的原生js实现 标签页(ta ...

  7. [CocoaPods]终端方式加载第三方库

    终端方式集成第三方库 1.打开终端,转到当前工程所在的文件夹. 方式一: [访达]->[服务]->[系统偏好设置] ->勾选[新建位于文件夹位置的终端标签 ]和[新建位于文件夹位置的 ...

  8. MySql数据保障

    1, 安装文档 配置文件,目录,参数,用户,权限,程序,安装方式 2, 数据备份 强大的备份策略,

  9. OpenStack-Ocata版+CentOS7.6 云平台环境搭建 — 3.安装配置OpenStack认证服务(keystone)

    节点配置信息说明: 控制节点:controller: IP:192.168.164.128 hostname&hosts:likeadmin 计算加点:Nova: IP:192.168.164 ...

  10. 手把手教你如何用eclipse搭建前端开发环境

    3.创建静态web工程 打开eclipse,选择file,new project 或者 new other...,选择web项中的static web project ,next. 输入你的项目名,如 ...