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. ...
随机推荐
- openssl知识点总结
openssl知识点总结 实践总结见之前博客:http://www.cnblogs.com/Jclemo/p/6091201.html 简介 openssl是一个功能丰富且自包含的开源安全工具箱.它提 ...
- 分享我收集的引擎、图形学、WebGL方面的电子资料
本文分享我这一年以来收集的我认为比较经典的电子资料,希望能对大家有所帮助! 本文会不断更新! 目录 WebGL Insights OpenGL Insights Game Programming Pa ...
- js判断页面出现滚动条
当可视区域小于页面的实际高度时,判定为出现滚动条 当我们在获取页面的offsetHeight高度时是包括了浏览器的边框的,浏览器的边框是2个像素, if (document.documentEleme ...
- vs操作快捷键
注释: 先CTRL+K,然后CTRL+C取消注释: 先CTRL+K,然后CTRL+U 解析命名空间:shift+alt+f10 或Ctrl + . 调试快捷键 F6: ...
- 08.C# System.Nulable<T>和空引用操作符(四章4.2-4.4)
看了这3小节,发现作者讲得太详细了,把一个都在正常使用的用法说得太神密了,搞得不知是自己不懂作者的苦心,还是作者用意为之,这里给大家都简单讲下吧,太深的真心讲不下去. 1.可空类型的核心部分是Syst ...
- WCF 入门(23,24)
前言 重新找了工作,过去的事情就过去好了.有些不一样的经历也不是说一定就是坏事. 第24集 在控制台应用程序中自托管WCF服务 Self hosting a wcf service in consol ...
- 第二十五课:jQuery.event.trigger的源码解读
本课主要来讲解jQuery.event.trigger的源码解读. trigger = function(event, data, elem, onlyHandlers){ if(elem & ...
- Moqui学习Day1
运行时目录以及Moqui的xml配置文件 Moqui框架部署运行主要有三个核心部分: 可执行的war包文件 运行时目录 Moqui配置文件XML格式 不管怎么使用这个可执行的war文件,你必须拥有一个 ...
- Linux svn 回滚版本库
Linux代码 svn up Index/ 然后找出要撤销的确切版本: Linux代码 svn log --limit 10 Index/tpl/css/global.css 根据log怀疑是 ...
- 每天一个linux命令(6):mv命令
mv命令是move的缩写,可以用来移动文件或者将文件改名(move (rename) files),是Linux系统下常用的命令,经常用来备份文件或者目录. 1.命令格式: mv [选项] 源文件或目 ...