C# 多线程通信详解
一、WaitHandler的类层次

可以看到 WaitHandle是 事件(EventWaitHandle)、互斥体(Mutex)、信号量(Sempahore)的父类。


WaitHandle我们最经常使用的方法,并是使用它的静态方法WaitAll. 我们会发现在这个WaitHandle里面只有等待方法,也就是它会阻塞当前线程的执行。
那么如何要解除它对当前线程的阻塞呢,那么就需要依赖于各个子类的方法了。



例如现在有一个这样的场景,如何在一个方法中,等待所有的线程全部执行完,最后再统计得到的计算结果呢?
WaitHandle[] handlers = new WaitHandle[]{
new AutoResetEvent(false),
new AutoResetEvent(false),
new AutoResetEvent(false),
new AutoResetEvent(false),
new AutoResetEvent(false),
new AutoResetEvent(false),
new AutoResetEvent(false),
new AutoResetEvent(false)
};
for (var i = 0; i < handlers.Length; i++)
{
ThreadPool.QueueUserWorkItem(ar =>
{
int index = (int)ar;
Thread.Sleep(1000);
AppCenter.AppendLog("任务:" + index + "开始执行!");
(handlers[index] as AutoResetEvent).Set();
}, i);
}
ThreadPool.QueueUserWorkItem(ar =>
{
WaitHandle.WaitAll(handlers);
AppCenter.AppendLog("所有任务都已经完成了,我不用再等待了。");
});
运行结果如下:

二、EventWaitHandle

这个方法,可以方便实现两个线程之间的相互通信。
如何实现两个线程的相互通信?
EventWaitHandle handleA = new AutoResetEvent(false);
EventWaitHandle handleB = new AutoResetEvent(false); ThreadPool.QueueUserWorkItem(ar =>
{
AppCenter.AppendLog("A:我是A,我已经开始运行了");
Thread.Sleep(2000);
AppCenter.AppendLog("A:我想睡觉了,B你先跑跑吧。");
EventWaitHandle.SignalAndWait(handleB, handleA);
AppCenter.AppendLog("A:开始工作ing");
Thread.Sleep(3000);
AppCenter.AppendLog("A:这个有点难,问下B");
EventWaitHandle.SignalAndWait(handleB, handleA);
AppCenter.AppendLog("A:不错,今天任务搞定,我也闪人了。");
}); ThreadPool.QueueUserWorkItem(ar =>
{
handleB.WaitOne();
AppCenter.AppendLog("B:我是B,我已经顶替A开始运行了。");
Thread.Sleep(5000);
AppCenter.AppendLog("B:我的事情已经做完了,该让A搞搞了,休息一会。");
EventWaitHandle.SignalAndWait(handleA, handleB);
AppCenter.AppendLog("B:hi,A我搞定了,下班了。");
handleA.Set();
});
运行结果如下:

那么AutoResetEvent和ManualResetEvent有什么区别呢?我们先做个实验。
private EventWaitHandle manualEvent = new ManualResetEvent(false); private void ManualResetEvent_Click(object sender, EventArgs e)
{ AppCenter.CleanLogs(); ThreadPool.QueueUserWorkItem(ar =>
{
int i = 0;
while (true)
{
manualEvent.WaitOne(); //ManualResetEvent的Set()方法,让事件的终止状态永远为true,让这里一直能执行。
i++; //而AutoResetEvent的Set()方法,初始化让这里执行一次,然后再次执行时是非终止的。将阻塞原有线程的执行
AppCenter.AppendLog("#" + i.ToString());
Thread.Sleep(1000);
}
});
} private void button2_Click(object sender, EventArgs e)
{
manualEvent.Set();
} private void button3_Click(object sender, EventArgs e)
{
manualEvent.Reset();
}
运行结果

我们会发现ManualResetEvent在触发Set()方法会,解除了原有的线程的 WaitOne方法,会一直打印输出。
而当我们替换为AutoResetEvent方法时候。

此时每次只会打印一个输出。因为它将 事件的状态设置为终止后,又变为了false.
三、Semaphore 控制并行线程的执行
应用场景,如果有多个线程跑,我能否每次控制3个线程一起跑呢。
Semaphore sempore = new Semaphore(0, 3);
for (int i = 0; i < 8; i++)
{
ThreadPool.QueueUserWorkItem(ar =>
{
sempore.WaitOne();
AppCenter.AppendLog("\t第:" +((int)ar).ToString() + "个开始运行.");
},i);
}
ThreadPool.QueueUserWorkItem(ar =>
{
for (int i = 0; i < 3; i++)
{
AppCenter.AppendLog("第" + (i + 1).ToString() + "批开始执行.");
sempore.Release(3);
Thread.Sleep(5000);
}
});
运行结果:

C# 多线程通信详解的更多相关文章
- Android中实现java与PHP服务器(基于新浪云免费云平台)http通信详解
Android中实现java与PHP服务器(基于新浪云免费云平台)http通信详解 (本文转自: http://blog.csdn.net/yinhaide/article/details/44756 ...
- .NET多线程同步方法详解
.NET多线程同步方法详解(一):自由锁(InterLocked) .NET多线程同步方法详解(二):互斥锁(lock) NET多线程同步方法详解(三):读写锁(ReadWriteLock) .NET ...
- Android Socket通信详解
一.Socket通信简介 Android与服务器的通信方式主要有两种,一是Http通信,一是Socket通信.两者的最大差异在于,http连接使用的是“请求—响应方式”,即在请求时建立连接通道,当客 ...
- SSL握手通信详解及linux下c/c++ SSL Socket代码举例
SSL握手通信详解及linux下c/c++ SSL Socket代码举例 摘自:http://www.169it.com/article/3215130236.html 分享到:8 发布时 ...
- SSL握手通信详解及linux下c/c++ SSL Socket代码举例(另附SSL双向认证客户端代码)
SSL握手通信详解及linux下c/c++ SSL Socket代码举例(另附SSL双向认证客户端代码) 摘自: https://blog.csdn.net/sjin_1314/article/det ...
- Java多线程——多线程方法详解
本系列文章是Java多线程的详解介绍,对多线程还不熟悉的同学可以先去看一下我的这篇博客Java基础系列3:多线程超详细总结,这篇博客从宏观层面介绍了多线程的整体概况,接下来的几篇文章是对多线程的深入剖 ...
- 开源项目SMSS发开指南(五)——SSL/TLS加密通信详解(下)
继上一篇介绍如何在多种语言之间使用SSL加密通信,今天我们关注Java端的证书创建以及支持SSL的NioSocket服务端开发.完整源码 一.创建keystore文件 网上大多数是通过jdk命令创建秘 ...
- java网络编程Socket通信详解
Java最初是作为网络编程语言出现的,其对网络提供了高度的支持,使得客户端和服务器的沟通变成了现实,而在网络编程中,使用最多的就是Socket.像大家熟悉的QQ.MSN都使用了Socket相关的技术. ...
- Swift - 多线程GCD详解
// GCD详解 // 目录: // 1. 创建GCD队列(最常用) // 2. 自定义创建队列 // 3. 使用多线程实现延迟加载 // 4. 使用多线程实现重复(循环) // 5. ...
随机推荐
- 三种实例化委托的方式(C# 编程指南)
1.定义的委托和方法 delegate void TestDelegate(string s); static void M(string s) { Console.WriteLine(s); } 2 ...
- Ubuntu安装uget和aria2下载工具
Windows下的下载工具有迅雷,快车等.Ubuntu下也有强大的下载工具uget和aria2. 一.安装.uget和aria2都可以在“软件中心”中安装,但是版本太老啦,所以最好还是在终端中添加pp ...
- HTML5 IE兼容问题
最近,为公司做产品的时候用到了HTML5,用IE11打开的时候出现了界面乱或者加载js错误的问题. IE10 or IE11 :Browser Mode is IE10 .Document Mode: ...
- 慢牛APP相关截图
慢牛APP相关截图 第一和第二个版本都是基于Sencha Touch+Cordova开发,公众号是采用Angularjs+D3开发,第三个版本是采用React Native开发. 第一个版本 第二个版 ...
- 最简单的 Web Service 入门 (看了包会)
原理:WebService是一个SOA(面向服务的编程)的架构,它是不依赖于语言,不依赖于平台,可以实现不同的语言间的相互调用,通过Internet进行基于SOAP协议的网络应用间的交互. 作用:主要 ...
- 【转载】gcc 使用中常用的参数及命令
本文转载自:http://www.cnblogs.com/yaozhongxiao/archive/2012/03/16/2400473.html 如需转载,请注明原始出处.谢谢. --------- ...
- KMP和扩展KMP
文章网上太多这里提一下代码细节: KMP: scanf("%s\n",s); scanf("%s\n",t); int ls=strlen(s),lt=strl ...
- OC基础--OC中类的定义
OC中类的定义与使用跟C#和java差别相当明显,做个笔记,牢记并加以区别! 一.OC中类的定义:关键字@implementation 和 @end 注意事项: 1.定义好一个类之后,要让这个类继承N ...
- jeecms内容显示条数
1.按照1.2.3.4.5顺序显示 <div class="index-news"> [@cms_channel id='1'] <h2><span& ...
- 【BZOJ-1449&2895】球队收益&球队预算 最小费用最大流
1449: [JSOI2009]球队收益 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 648 Solved: 364[Submit][Status][ ...