.NET多线程知识快速学习
多线程是一个不会过时的话题,因为每个开发的成长必然要掌握这个知识点,否则半懂不懂怎么保证系统的可靠性和性能,其实在网上随便一搜都会有海量的文章说这个话题,大多数写得很细写得非常好,但发现很少有概览性的文章,我希望能借本文给大家一个全局视野,结合多年实践帮大家快速学习或者回顾思考,对感兴趣的知识点再深入学习了解。
一、知识点概括

二、具体实例演示如何实现
> Thread 多线程最基础的类
//ThreadPool.GetMaxThreads(out int maxTCount, out int maxPCount);
//ThreadPool.GetMinThreads(out int minTCount, out int minPCount);
//ThreadPool.SetMaxThreads(maxTCount, maxPCount);// 调整最大线程数
//ThreadPool.SetMinThreads(minTCount, minPCount);// 调整最小线程数 long tick = C_ITEM_COUNT;
ManualResetEvent signal = new ManualResetEvent(false);
Console.WriteLine("========== 示例:采用Thread执行处理 ==========");
for (int i = ; i < C_ITEM_COUNT; i++)
{
new Thread((obj) =>
{
Thread.Sleep();
Console.Write(" {0} ", obj);
if (Interlocked.Decrement(ref tick) == )
signal.Set();
}).Start(i);
}
Console.Write(" 等待子线程执行 ");
signal.WaitOne();
Console.WriteLine();
Console.WriteLine("全部线程执行完毕,按任意键继续...");
> ThreadPool 线程池,XX池的概念可以广泛应用于其他资源管理,例如字体池(防句柄泄露)、短信猫池等等
tick = C_ITEM_COUNT;
signal.Reset();
Console.WriteLine();
Console.WriteLine("========== 示例:采用ThreadPool执行处理 ==========");
for (int i = ; i < C_ITEM_COUNT; i++)
{
ThreadPool.QueueUserWorkItem((obj) =>
{
Thread.Sleep();
Console.Write(" {0} ", obj);
if (Interlocked.Decrement(ref tick) == )
signal.Set();
}, i);
}
Console.Write(" 等待子线程执行 ");
signal.WaitOne();
Console.WriteLine();
Console.WriteLine("全部线程执行完毕,按任意键继续...");
> Task 任务,功能丰富用法灵活。结合现实生活用字面意思去理解就好:可以同时做多个任务,任务做完可以接着做其他任务,任务可能会取消等等。
tick = C_ITEM_COUNT;
signal.Reset();
Console.WriteLine();
Console.WriteLine("========== 示例:采用Task执行处理,注意取消了处理{0}的进程 ==========", C_ITEM_COUNT - );
var tasks = new Tuple<Task, CancellationTokenSource>[C_ITEM_COUNT];
for (int i = ; i < C_ITEM_COUNT; i++)
{
var cts = new CancellationTokenSource();
var task = Task.Factory.StartNew((obj) =>
{
Thread.Sleep();
Console.Write(" {0} ", obj);
}, i, cts.Token);
task.ContinueWith((t) =>
{
if (Interlocked.Decrement(ref tick) == )
signal.Set();
});
tasks[i] = new Tuple<Task, CancellationTokenSource>(task, cts);
} tasks[C_ITEM_COUNT - ].Item2.Cancel();// 取消线程。 Console.Write(" 等待子线程执行 ");
signal.WaitOne();
Console.WriteLine();
Console.WriteLine("全部线程执行完毕,按任意键继续...");
> Parallel 并行
tick = C_ITEM_COUNT;
signal.Reset();
Console.WriteLine();
Console.WriteLine("========== 示例:采用Parallel执行处理 ==========");
Parallel.For(, C_ITEM_COUNT, obj =>
{
Thread.Sleep();
Console.Write(" {0} ", obj);
if (Interlocked.Decrement(ref tick) == )
signal.Set();
}); Console.Write(" 等待子线程执行 ");
signal.WaitOne();
Console.WriteLine();
Console.WriteLine("全部线程执行完毕,按任意键继续...");
以上示例执行结果如下,重点可以关注下"等待子线程执行"这个节点,理解主线程和各子线程的优先执行顺序

三、性能对比(理解线程池技术的性能也可以通过最大最小线程数调节)
> 循环数:200,线程池参数:默认

> 循环数:200,线程池参数:50 - 1000

> 循环数200,线程池参数:100-1000

> 循环数200,线程池参数:200-1000

| 最大线程数 ~ 最小线程数 | Thread(ms) | ThreadPool(ms) | Task(ms) | Parallel(ms) |
| 2047/1000 ~ 12/12 | 2712.09 | 8057.14 | 8585.01 | 7526.57 |
| 1000/1000 ~ 50/50 | 2733.25 | 2289.96 | 2218.29 | 3660.33 |
| 1000/1000 ~ 100/100 | 2503.08 | 1620.73 | 1534.50 | 1742.78 |
| 1000/1000 ~ 200/200 | 2999.27 | 1436.24 | 1150.21 | 935.22 |
四、结论
> Thread就像脱缰的野马,不受控制,创建多少就运行多少,可能少量时效率是高了,量大的时候除了性能没优势,还可能导致句柄泄露。
> ThreadPool与Task类似,但Task相比效率更高用法更灵活。
> Parallel自带了同步功能,不需要用信号量来做额外的同步等待。
> ThreadPool、Task、Parallel的性能都取决于线程池最大线程数和最小线程数。
> 推荐使用 Task 和 Parallel,具体用哪个可以参考用法自己斟酌。
.NET多线程知识快速学习的更多相关文章
- 第23篇 js快速学习知识
前面说了js的一些高级方面的基础知识,这些都是比较容易出错的和比较难理解的东西,除了这些之外其它的知识都比较简单了,基础学好了,扩展起来就是小意思.今天说说js方面可以快速学习和入门的知识. 1.闭包 ...
- 看我如何快速学习.Net(高可用数据采集平台)
最近文章:高可用数据采集平台(如何玩转3门语言php+.net+aauto).高并发数据采集的架构应用(Redis的应用) 项目文档:关键词匹配项目深入研究(二)- 分表思想的引入 吐槽:本人也是非常 ...
- 阿里巴巴Java开发手册快速学习
Java作为一门名副其实的工业级语言,语法友好,学习简单,大规模的应用给代码质量的管控带来了困难,特别是团队开发中,开发过程中的规范会直接影响最终项目的稳定性. 善医者“未有形而除之”,提高工程健壮性 ...
- Javaweb快速学习
孙卫琴老师的javaweb一书已经买了很多年,由于很厚一直也没有去好好阅读下, 项目发布后有闲暇时间,决定快速学习了,毕竟很多概念和知识主要还是复习. 对于互联网,我们可以简单认为浏览器就是会人类语言 ...
- Vue 超快速学习
Vue 超快速学习 基础知识: 1.vue的生命周期: beforeCreate/created. beforeMount/mounted. beforeUpdate/updated. beforeD ...
- 20155321 《信息安全系统设计》Linux多线程的深入学习
再次学习之多线程 基本概念的再次学习 线程是程序执行的最小单位(进程是资源管理的最小单位),线程隶属于某个进程中 进程有自己的数据段.代码段和堆栈段.线程通常叫做轻型的进程,每个线程共享其所附属进程的 ...
- 一点记忆—— Java框架知识的学习有感
一点记忆—— Java框架知识的学习有感 说一下自己学习的框架知识的历程吧:好久了,应该是一年前,也就是大三上的时候,我对于基本的 Java编程就已经达到所谓的“熟练”,认为自己可以在 ...
- (*)(转)要快速学习SSM框架,你需要一套学习曲线平滑的教程
作者:meepo链接:https://www.zhihu.com/question/57719761/answer/156952139来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载 ...
- 【原创】docker在Ubuntu下1小时快速学习
前言 由于工作原因,很多情况下需要快速学习新的知识,针对docker如果从头到尾看相关书籍学习会非常慢,所以整理了下docker的常用操作,只要跟着本文学习操作,一小时就能掌握docker大部最常用分 ...
随机推荐
- jquery查出元素名称
<div onclick='$(this).prop("tagName")'></div> 最后的结果 DIV
- QT使用mysql
1.首先要下载qt create 官网链接:https://wiki.qt.io/Main 2.下载mysql驱动mysql-connector-c,注意是c或c++版本的驱动 官网下载地址:http ...
- VS2017 无法修改代码编辑区的项背景颜色问题
以前都是好好的,安装 ClaudiaIDE: https://github.com/buchizo/ClaudiaIDE 之后也没啥问题,用着用着代码编辑区自定义的颜色就没有了,好几台电脑都是这样, ...
- 计划任务cron
cron 计划任务 作用: 计划任务主要是做一些周期性的任务,目前最主要的用途是定期备份数据 Schedule one-time tasks with at. 一次性调度执行 atSchedule r ...
- Nmap参数详解(含扫描参数原理解释)
语法结构:nmap [Scan Type(s)] [Options] {target specification} 端口状态介绍 open:确定端口开放,可达 closed :关闭的端口对于nmap也 ...
- .Net Core HTTP Error 500.0 - ANCM In-Process Handler Load Failure 发布后启动错误处理
方法1: 应用池高级配置 - 启动32位应用程序 - 设置为True 方法2: web.config <aspNetCore processPath="%LAUNCHER_PATH%& ...
- 如何实现一台服务器同时运行两个php版本
有需要学习交流的友人请加入交流群的咱们一起,有问题一起交流,一起进步!前提是你是学技术的.感谢阅读! 点此加入该群jq.qq.com 假设您已经安装了Apache,为这两个项目创建了虚拟主机,并添加 ...
- Spring Cloud Finchley.SR1 版本的坑:placeholer占位符无法解析!
接入nacos 之后,想把所有的配置丢上去. 启动程序是: @EnableDiscoveryClient @RestController @ComponentScan(basePackages = { ...
- IT兄弟连 HTML5教程 DIV+CSS网站首页布局示例
首页的设计直接影响网站的整体形象,虽然没有一个统一的规范,但最好将其设计为大众化的,只要信息内容能够合理地编排即可,使用户可以方便地找到需要的信息.另外,首页的高度最好不要超过三个屏幕,页面中使用的颜 ...
- C#中转换运算符explicit、implicit、operator、volatile研究
C#中的这个几个关键字:explicit.implicit与operator,估计好多人的用不上,什么情况,这是什么?字面解释:explicit:清楚明白的;易于理解的;(说话)清晰的,明确的;直言的 ...