本篇概述:

@FCL4.0中已经存在的线程模型,以及它们之间异同点;

@多线程编程模型的选择。

1:异步、多线程、任务、并行的本质

这四个概念对应在CLR中的本质,本质都是多线程。

异步,简单的讲就是BeginInvoke、EndInvoke模式,它在CLR内部线程池进行管理;

多线程,体现在C#中,可以由类型Thread发起。也可以由ThreadPool发起。前者不受CLR线程池管理,后者则是。FCL团队为了各种编程模型的方便,还另外提供了BackgroundWorker和若干个Timer,基本上它们都是ThreadPool的加强,增加了一些和调用者线程的交互功能;

任务(Task),为FCL4.0(框架类库(Framework Class Library,FCL),)新增的功能,在一个称之为任务并行库(TPL)的地方,其实也就是System.Threading.Tasks命名空间下。任务并行库名字取的很玄乎,其实它也是CLR线程池的加强。优化了线程间的调度算法,增加了和调用者线程的交互功能;

并行(Parallel),为FCL4.0新增的功能,也属于TPL。并行在后台使用Task进行管理,说白了,因为Task使用的线程池线程,所以Parallel自然使用的也是线程池线程进行管理,它的本质仅仅是进一步简化了Task。在这里要增进一个对于并行的理解。实际上,多线程天然就是并行的。及时不用任务并行库,用Thread类型新起两个线程,CLR或者说Windows系统也会将这两个线程根据需要安排到两个CPU上去执行。所以,并不是因为多了任务并行库,CLR才支持并行计算,任务并行库只是提供了一组API,使我们能够更好的操纵线程进行并行开发而已。

2:遗憾

Jeffrey Richter大叔说,微软提供了这么多线程模型,是遗憾的,因为这制造了混乱。很多开发者都不知道该选用哪个类型来编写自己的多线程代码。我们对微软总是又爱又恨,它总是不停的更新一些东西,逼迫我们不停的学习。但是也好,进步导致它不会过早死掉,让我们彻底失掉饭碗。
C#刚出来的被人笑,现在它的很多语法特性已经比Java优美。很多时候我们太擅长于嘲笑,以致最后只能哭。顺便说一句,我依然是那么的喜欢JAVA,只是很久没用它而已。

3:现在,该用什么来编写多线程

如果你在FRAMEWORK4.0下编写代码,那么应该按照这个优先级来撰写多线程代码:

优先

次优先

不得以

Parallel(含扩展库PLinq)

Task

ThreadPool(BackgroundWorker,Timer)

异步

Thread

这个表满足了大部分情况下的一个优先级指导,但在某些情况下会有例外。

3.1:为什么 Parallel和Task优先级一样,而不是优于Task?

Parallel虽然在后台使用Task进行管理,并且它所谓简化了对于Task的操作,但是它有一个重要的特征区别与Task:Parallel会阻滞调用者线程。查看Paralle的成员,有For、ForEach、Invoke方法,它甚至都没提供一个BeginInvoke方法,也很好的暗示了这一点。不过虽然是同步的执行的,Parallel还是会把多个任务分配到多个CPU上去。
Task被用的最多的是Start方法,它不会阻滞主线程。虽然Task也提供了同步的启动线程的方法RunSynchronously,但一般用的不多。

3.2:何时用异步,何时用线程或线程池

这需要从“IO操作的DMA(Direct Memory Access)模式”讲起。通过DMA的数据交换几乎可以不损耗CPU的资源。在硬件部分,硬盘、网卡、声卡、显卡等都有DMA功能。可以简单的认为,当我们的工作线程需要操作I/O资源的时候(如读取一个大文件、读取一个网页、读取Socke包等),我们就需要用异步去做这些事情。异步模式只会在工作开始以及工作结束的时候占用CLR线程池,其它时候由硬盘、网卡等硬件设备来处理具体的工作,这就不会过多占用到CPU空间和时间损耗。

概括而言:

计算密集型工作,直接采用线程;

IO密集型工作,采用异步机制;

当我们不清楚什么工作是I/O密集型的,一个不是很恰当的指导就是:查看FCL类型成员,如果成员提供了类似BeginDosomething方法的,则优先使用它,而不是新起一个线程或丢到线程池。

3.3:线程池的优势

新起线程,会带来很大的开销,这些开销主要集中在:分配线程内核对象、线程环境块、用户模式栈、内核模式栈所需要的内存空间,加载的DLL的DLLMain方法,并传递连接标志,以及线程上下文切换。由于线程如此昂贵,所以对于普通的开发要求来说,线程池就是一个很好的选择。线程池替开发人员管理工作线程,当一项工作完毕的时候,CLR不会销毁这个线程,而是会保留这个线程一段时间,看是否有别的工作需要这个线程。至于何时销毁或新起线程,由CLR决定。

3.4:何时用Thread 
以上的各种线程模型,它们最终都是Thread。 那么什么时候需要Thread直接出场呢?

最重要的使用Thread的理由是,我们需要控制线程的优先级。Thread之上的线程模型都不支持优先级设置。设置一个线程的高优先级可以使它获得更多的CPU时间;

再者,可以控制线程为前台线程。当然,由Thread新起的线程默认就是前台线程。前台线程不随着调用者线程的中断而中断,这使得我们可以用Thread来进行一些关键性的操作。

 

本文基于Creative Commons Attribution 2.5 China Mainland License发布,欢迎转载,演绎或用于商业目的,但是必须保留本文的署名http://www.cnblogs.com/luminji(包含链接)。如您有任何疑问或者授权方面的协商,请给我留言。

异步、多线程、任务、并行编程之一:选择合适的多线程模型相关推荐

  1. java 多线程和并行编程_Java 8中的并行和异步编程

    java 多线程和并行编程 并行代码是在多个线程上运行的代码,曾经是许多经验丰富的开发人员的噩梦,但是Java 8带来了许多更改,这些更改应该使这种提高性能的窍门更加易于管理. 并行流 在Java 8 ...

  2. C#多线程与并行编程方面的电子书,中英文版本

    给大家共享一些C#多线程与并行编程方面的电子书,中英文版本的. 链接: 百度网盘地址 提取码: y99a 转载于:https://www.cnblogs.com/rainbow70626/p/1069 ...

  3. 在实际项目中,如何选择合适的机器学习模型?

    https://blog.csdn.net/gitchat/article/details/78913235 本文来自作者 chen_h 在 GitChat 上分享 「在实际项目中,如何选择合适的机器 ...

  4. 面对数据缺失,如何选择合适的机器学习模型?

    作者:阿萨姆 普华永道 | 数据科学家 量子位 已获授权编辑发布 转载请联系原作者 有些小伙伴在实际使用中发现xgboost可以自动处理缺失值,而有些模型不可以.我想先从两个角度解答这个困惑: 工具包 ...

  5. Linux多线程服务端编程笔记,陈硕Linux多线程服务端编程读书笔记 —— 第一章 线程安全的对象生命周期管理...

    muduo书第一章的思维导图(新窗口打开可以看大图) 线程安全的对象析构和回调(使用shared_ptr和weak_ptr) 注释是英文的(工地英语-) StockFactory.h // in mu ...

  6. C#中的多线程 - 并行编程 z

    原文:http://www.albahari.com/threading/part5.aspx 专题:C#中的多线程 1并行编程Permalink 在这一部分,我们讨论 Framework 4.0 加 ...

  7. JVM是如何进行多线程并行编程的

    灵魂画风小剧场镇楼! 我们在日常项目中会经会遇到希望用多线程来并行执行任务的场景.一方面,多线程可以提升执行效率,但同时它也增加了排查问题的难度. 其实,HotSpotVM里就提供了管理多线程的框架, ...

  8. Java 多线程与并发编程专题

    Java 线程基础 Java 多线程开发 线程安全与同步 并发控制 非阻塞套接字(NIO) Java 5 中的并发 JDK 7 中的 Fork/Join 模式 相关书评 Java 平台提供了一套广泛而 ...

  9. 如何选择合适的太阳能光伏浪涌保护器

    您是否担心您昂贵的太阳能光伏系统有一天会遇到雷击电涌而损坏? 目前现实情况是,大多数光伏和太阳能系统都忽视防雷保护,很多雷击频繁的地区设备没有安装浪涌保护器,过载电涌和电压是危险的,即使是最轻微的电压 ...

最新文章

  1. Nginx笔记系列(1)——Nignx的安装部署
  2. OpenGL之矩阵堆栈绘制立体图元
  3. Flume-NG源码阅读之AvroSink
  4. weex安装环境_WEEX跨平台开发环境搭建
  5. 前端学习(654):优化插件
  6. 服务端/web+客户端模式配置
  7. hdfs中与file数组类似的数组_EXCEL中数组的应用专题之十二:行列数相同数组的运算...
  8. ASP.NET MVC 使用 Log4net 记录日志
  9. macappstore登不上去_丈夫在外与多人暧昧,妻子气愤不已,可他却怒怼:这还不都是你逼的...
  10. MySQL DEBUG_SYNC 的简单分析与测试
  11. AI公开课丨胡伟:基于表示学习的知识图谱实体对齐研究
  12. 执行mount命令时找不到介质或者mount:no medium found的解决办法
  13. MongoDB(芒果数据库)学习(一)———增删改查
  14. 高德 各省市区县poi及区划名称 代码/高德历史poi
  15. centos查询 硬盘序列号查询_CentOS/Linux 查看硬件配置CPU内存磁盘
  16. 量化交易学习——熟读github上的开源交易策略框架
  17. JAVA微信小程序医院预约挂号小程序系统毕业设计 开题报告
  18. Whole Word Masking (wwm) BERT PaddlePaddle常用预训练模型加载
  19. 附录H-2 技术预研报告
  20. 一道google面试题--自然数e中出现的连续的第一个10个数字组成的质数

热门文章

  1. python二进制、字符编码及文件操作
  2. Vector的使用详解
  3. 在京东上买的手机然后申请退货的经历
  4. 个人作业5——软件工程总结
  5. [转]ASP.NET MVC4中@model使用多个类型实例的方法
  6. Oracle解析XML 节点信息并转换String 类型
  7. steelray project viewer
  8. POJ 1860 Currency Exchange (SPFA松弛)
  9. Android--Facebook Login without LoginButton
  10. PHP面向对象2之变量、方法