上一章我们了解到,由于线程的创建,销毁都是需要耗费大量资源和时间的,开发者应该非常节约的使用线程资源。最好的办法是使用线程池,线程池能够避免当前进程中大量的线程导致操作系统不停的进行线程切换,当线程数量到达了我们设置的上限,线程会自动排队等待,当线程资源可用时,队列中的线程任务会依次执行,如果没有排队等候的资源,线程会变为闲置状态。

使用ThreadPool来访问线程池

这种做法可以让我们不用那么复杂的去实现创建,重用线程的逻辑,但是也有一些限制,比如通过内置的方法,我们无法知道什么时候线程池里面的任务会结束,也不能获取线程的返回值。为了解决这些问题,微软引入了一个新的概念。

使用Task来访问线程池

引入了Task之后,你可以用如下实现来替代ThreadPool

这三种方式的实现都是等价的。另外,Task本身实现了很多ThreadPool不能做的事情。

使用Task来获得线程的返回值

使用Task来等待线程结束

更多Task同步编程的使用,请参见(还没写,先给自己挖个坑O(∩_∩)O)。

异步委托

ThreadPool.QueueUserWorkItem没有提供一种简单的机制来获取线程的返回值。异步委托解决了这个问题,支持了传入一系列的参数。此外,异步委托中没有处理的异常会很方便的在调用线程的重新抛出(在调用EndInvoke的时候),因此不需要显式的处理。

通过异步委托来执行任务主要分一下几步:

  1. 初始化并声明一个你想要执行的委托
  2. 在委托上调用BeginInvoke,把返回值保存为IAsyncResult中

调用BeginInvoke不会阻塞当前线程,因此你可以在调用完之后执行其他你想要同步的操作

  1. 当你需要获取委托的返回值时,调用EndInvoke方法,把IAsyncResult传入EndInvoke中

阻塞的方式执行异步委托

EndInvoke主要做3件事: 1. 等待异步委托完成 2. 接收返回值 3. 把异步线程中未处理的异常在当前线程中重新抛出。

非阻塞的方式执行异步委托

你也可以在调用BeginInvoke的时候指定一个回调方法,这个方法会在异步委托结束的时候自动调用。这样异步委托就像是一个后台线程一样自动执行,不需要主线程等待。只需要在BeginInvoke的时候做一些额外的操作即可实现这种操作。

关于线程池

Jeffery在C# via CLR Chapter27中针对线程池的使用给出了一些建议。目前我们允许开发者来指定一个线程池的最大线程数。但是事实证明,我们往往不应该为一个线程池指定线程的上限,否则可能会出现程序死锁或者饿死的状态。比如你可能设置了1000个线程,但是某一时刻正好有第1001个线程需要等待所有线程结束才能执行,这种情况如果你限制了线程池线程的个数,就会出现死锁。从开发的另一个角度说,你也不应该限制一个进程使用多少资源,比如一个进程可以使用多少内存,使用多少带宽.因此虽然目前你可以通过GetMaxThreads, SetMaxThreads,GetMinThreads,SetMinThreads ,GetAvailableThreads来进行线程个数的限制,但是他仍然不建议大家这样做。这些限制可能会让你的程序运行的更慢。

关于使用Task访问线程池:

细说.NET中的多线程 (三 使用Task)

作者:独上高楼
出处:http://www.cnblogs.com/myprogram/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

【转】细说.NET中的多线程 (二 线程池)相关推荐

  1. java进阶之多线程二线程池

    单线程线程池 class threadPool{public static void main(String[] args) {long f = System.currentTimeMillis(); ...

  2. 【转】细说.NET 中的多线程 (一 概念)

    为什么使用多线程 1.使用户界面能够及时响应用户的输入 当某个应用程序在进行大量运算时候,为了保证应用程序能够随时响应客户的输入,这个时候我们往往需要让大量运算和响应用户输入这两个行为在不同的线程中进 ...

  3. 后端开发【一大波有用知识】Redis中的IO多线程(线程池)

    一.Redis中的IO多线程原理 ​服务端收到一条信息,给它deconde成一条命令 然后根据命令获得一个结果(reply) 然后将结果encode后,发送回去 redis的单线程是指,命令执行(lo ...

  4. Java基础巩固(二)异常,多线程,线程池,IO流,Properties集合,IO工具类,字符流,对象流,Stream,Lambda表达式

    一.异常,多线程 学习目标 : 异常的概述 异常的分类 异常的处理方式 自定义异常 多线程入门 1 异常的概述 1.1 什么是异常? 异常就是程序出现了不正常情况 , 程序在执行过程中 , 数据导致程 ...

  5. java同步锁售票_Java基础学习笔记: 多线程,线程池,同步锁(Lock,synchronized )(Thread类,ExecutorService ,Future类)(卖火车票案例)...

    学习多线程之前,我们先要了解几个关于多线程有关的概念. 进程:进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程序,并且具有一定独立功能. 线程:线程是 ...

  6. 多线程之线程池-各个参数的含义- 阿里,美团,京东面试题目

    阿里的面试官问了个问题,如果corepollSize=10,MaxPollSize=20,如果来了25个线程 怎么办, 答案: 当一个任务通过execute(Runnable)方法欲添加到线程池时: ...

  7. Java多线程之线程池配置合理线程数

    Java多线程之线程池配置合理线程数 目录 代码查看公司服务器或阿里云是几核的 合理线程数配置之CPU密集型 合理线程数配置之IO密集型 1. 代码查看公司服务器或阿里云是几核的 要合理配置线程数首先 ...

  8. 多线程编程—线程池的实现

    多线程编程-线程池的实现 执行与任务分离的组件- 线程池 https://github.com/wangbojing/threadpool 多线程技术主要解决了处理器单元内多个线程执行的问题,它可以显 ...

  9. C#多线程之线程池篇1

    在C#多线程之线程池篇中,我们将学习多线程访问共享资源的一些通用的技术,我们将学习到以下知识点: 在线程池中调用委托 在线程池中执行异步操作 线程池和并行度 实现取消选项 使用等待句柄和超时 使用计时 ...

最新文章

  1. Ajax中请求被缓存的问题
  2. 在jquery mobile制作app的几个页利用ajax将后台数据json数组动态加载到列表里面
  3. java.lang.IllegalStateException Unable to find a @SpringBootConfiguration错误解决方案(亲测)
  4. ubuntu linux下面运行《暗黑破坏神2》和英雄无敌3-死亡阴影
  5. vue项目没有启动成功的原因之一
  6. ehcache缓存的详细配置
  7. WinForm数据绑定--BindingContext
  8. android:ClassNotFoundException for Activity class的解决方法
  9. HDU1873 看病要排队【模拟+优先队列】
  10. 邮件整体解决方案_为什么推荐用EmailCamel来做Shopify邮件营销?
  11. 《Linux4.0设备驱动开发详解》笔记--第七章:Linux设备中的并发控制
  12. Keras指定GPU训练模式,设置GPU的使用量
  13. Java 技术书籍大全
  14. android仿微信选择器同时展示视频和图片
  15. Arcgis空间校正操作
  16. Linux ubuntu18.04下socket聊天室 私聊 群聊
  17. 无线电波的波段划分和应用
  18. Google Web Accelerator
  19. python鼠标监听_用Python监听鼠标和键盘事件
  20. ArcGIS Pro试用许可申请

热门文章

  1. python time模块
  2. android与php使用base64加密的字符串结果不一样解决方法
  3. Tcl 语言改写Java题目-1
  4. 水平+垂直布局-css (借鉴)
  5. TCL 中upvar 用法 (摘自http://www.cnblogs.com/kane1990/archive/2011/12/19/2293981.html)
  6. 弹出键盘,UIView 上移
  7. [转]宝文!Apple Push Notification Service (APNS)原理与实现方案
  8. [dp]leetcode 198. House Robber
  9. Oracle杀事务数据库崩溃,关于pl/sql dev窗口崩溃导致锁表
  10. css 背景图怎么设置自动填充满_CSS属性设置 -- 背景样式