热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:多线程之旅(ThreadPool线程池)

篇首语:本文由编程笔记#小编为大家整理,主要介绍了多线程之旅(ThreadPool线程池)相关的知识,希望对你有一定的参考价值。一、什么是

篇首语:本文由编程笔记#小编为大家整理,主要介绍了多线程之旅(ThreadPool 线程池)相关的知识,希望对你有一定的参考价值。


一、什么是ThreadPool 线程池(源码)

      1.线程池顾名思义,有我们的系统创建一个容器装载着我们的线程,由CLR控制的所有AppDomain共享。线程池可用于执行任务、发送工作项、处理异步 I/O、代表其他线程等待以及处理计时器。所以使用线程池不需要自己创建线程,而是通过线程池来创建和执行和管理线程。

二、ThreadPool 线程池和线程的区别

      1.ThreadPool 线程池是在.NET 2.0出现的,是一个享元模式整个程序共同享用这一个线程池,当我们的线程执行任务之后它不会立刻销毁,它会回到线程池中,如果有新的任务它就会去执行。避免了我们线程的重复创建和销毁(也不会造成我们CPU的上下文切换的损耗)。

      2.大家仔细看一下我前面写的Thread 创建线程执行任务之后,它会自动销毁。那问题来了我们经常的创建、销毁线程这可都是资源的浪费呀!!所以我们要利用每个线程占有的资源。

三、ThreadPool 线程池缺点

       1.线程池在性能上优于线程,但是它也是有缺点的。它不支持线程的取消、完成、失败通知等交互性操作。

       2.它不能设置池中线程的Name,会增加使用者的难度。

  3.线程池中线程通常都是后台线程,优先级为ThreadPriority.Normal

  4.线程池堵塞会影响我们的性能,阻塞会使CLR错误地认为它占用了大量CPU。CLR能够检测或补偿(往池中注入更多线程),但是这可能使线程池受到后续超负荷的印象。Task (也及时后面要讲的)解决了这个问题。

  5.线程池使用的是全局队列,全局队列中的线程依旧会存在竞争共享资源的情况,从而影响性能(Task 解决了这个问题,方案是使用本地队列)。

四、线程池工作原理

  1.CLR初始化时,线程池中是没有线程的,但是内部有一个操作请求队列,当我们的应用程序使用异步时,会将一个记录项添加到线程池的队列中,线程池队列会自动读取这个记录项,并且发给一个线程池的线程,如果线程池没有线程就会创建一个线程执行这任务,当线程完成任务它不会自动销毁而是回到我们的线程池中,等待线程池派发新的任务。

  2.如果程序给线程池派发了很多任务,线程池也会使用这一个线程执行所有的任务,如果我们的请求速度大于了我们的线程处理速度,就会创建额外线程,就不会导致我们创建了过多的线程。

  2.当我们的线程有大量休息的,它们会在一段时间内自动销毁。这样很好的控制了我们应用程序的性能。

五、ThreadPool 线程池使用

  1.ThreadPool是一个静态类,调用QueueUserWorkItem方法,是可以将一个异步计算放入我们的线程池队列中。


public static bool QueueUserWorkItem(WaitCallback callBack);
public static bool QueueUserWorkItem(WaitCallback callBack, object state);

































方法说明
QueueUserWorkItem启动线程池里的一个线程(工作者线程)
GetMinThreads检索线程池在新请求预测中能够按需创建的线程的最小数量。
GetMaxThreads最多可用线程数,所有大于此数目的请求将保持排队状态,直到线程池线程由空闲。
GetAvailableThreads剩余空闲线程数。
SetMaxThreads设置线程池中的最大线程数(请求数超过此值则进入队列)。
SetMinThreads设置线程池最少需要保留的线程数。

 

 

 

 

 

 

  2.我们可以看到ThreadPool比Thread少了很多的API,被砍掉了


技术图片技术图片

///


/// ThreadPool的使用
/// workerThreads  CLR线程池分为工作者线程(workerThreads)
/// completionPortThreads I/O线程(completionPortThreads)
///

public static void Show()
{
//使用线程
ThreadPool.QueueUserWorkItem((x) => Running());
ThreadPool.GetAvailableThreads(
out int workerThreads, out int completionPortThreads);
Console.WriteLine($
"没有设置线程数之前 workerThreads:{workerThreads} completionPortThreads:{completionPortThreads}");
//设置最大的线程数
ThreadPool.SetMaxThreads(16, 16);
//设置最小的线程数
ThreadPool.SetMinThreads(8, 8);
ThreadPool.GetAvailableThreads(
out int workerThreads1, out int completionPortThreads1);
Console.WriteLine($
"设置最大的线程数之后 workerThreads:{workerThreads1} completionPortThreads:{completionPortThreads1}");
Console.ReadLine();
}


View Code

技术图片

   3.我们要注意的就是堵塞线程的时候一定要做好处理,最好是不要堵塞我们的线程,不然很容易造成死锁GG


技术图片技术图片

///


/// ThreadPool 线程等待
///类 包含了一个bool属性
///false--WaitOne等待--Set--true--WaitOne直接过去
///true--WaitOne直接过去--ReSet--false--WaitOne等待
///https://www.cnblogs.com/howtrace/p/11362284.html
///

public static void Show1()
{
//设置最大的线程数
ThreadPool.SetMaxThreads(16, 16);
//设置最小的线程数
ThreadPool.SetMinThreads(8, 8);
//设置false使用WaitOne()会直接堵塞线程,不会释放 、Set()设置为true
ManualResetEvent manualResetEvent = new ManualResetEvent(false);
//设置false使用WaitOne()会直接堵塞线程,不会释放 、Set()设置为true
AutoResetEvent autoResetEvent = new AutoResetEvent(false);
//上面两种方法都是可以拦截线程,都是继承EventWaitHandle 接口
//就都具有Reset() //红灯 设置为false导致线程等待
//Set() //绿灯 设置为true 启动线程继续执行
//WaitOne() // 等待信号 会根据我们线程状态执行,为true不需要等待直接执行
//反之为false会等待线程状态为true才会执行
//不同点 ManualResetEvent AutoResetEvent
//ManualResetEvent 在·使用Set()的时候会所有处理 WaitOne 状态线程均继续执行。
//AutoResetEvent 在使用Set()的时候会执行一个线程其他的线程继续等待执行。
for (int i = 0; i <20; i++)
{
var k = i;
ThreadPool.QueueUserWorkItem(x
=>
{
Console.WriteLine(k);
if (k <18)
{
//等待线程,但是上面我们只开了16个线程,结果我18个线程全部等待
//导致了死锁
manualResetEvent.WaitOne();
}
else
{
//恢复执行状态
manualResetEvent.Set();
}
});
if (manualResetEvent.WaitOne())
{
Console.WriteLine(
"没有死锁、、、");
}
Console.WriteLine(
"等着QueueUserWorkItem完成后才执行");
}
Console.ReadLine();
}


View Code

推荐阅读
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 重入锁(ReentrantLock)学习及实现原理
    本文介绍了重入锁(ReentrantLock)的学习及实现原理。在学习synchronized的基础上,重入锁提供了更多的灵活性和功能。文章详细介绍了重入锁的特性、使用方法和实现原理,并提供了类图和测试代码供读者参考。重入锁支持重入和公平与非公平两种实现方式,通过对比和分析,读者可以更好地理解和应用重入锁。 ... [详细]
  • STL迭代器的种类及其功能介绍
    本文介绍了标准模板库(STL)定义的五种迭代器的种类和功能。通过图表展示了这几种迭代器之间的关系,并详细描述了各个迭代器的功能和使用方法。其中,输入迭代器用于从容器中读取元素,输出迭代器用于向容器中写入元素,正向迭代器是输入迭代器和输出迭代器的组合。本文的目的是帮助读者更好地理解STL迭代器的使用方法和特点。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • LeetCode笔记:剑指Offer 41. 数据流中的中位数(Java、堆、优先队列、知识点)
    本文介绍了LeetCode剑指Offer 41题的解题思路和代码实现,主要涉及了Java中的优先队列和堆排序的知识点。优先队列是Queue接口的实现,可以对其中的元素进行排序,采用小顶堆的方式进行排序。本文还介绍了Java中queue的offer、poll、add、remove、element、peek等方法的区别和用法。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • 深入理解Kafka服务端请求队列中请求的处理
    本文深入分析了Kafka服务端请求队列中请求的处理过程,详细介绍了请求的封装和放入请求队列的过程,以及处理请求的线程池的创建和容量设置。通过场景分析、图示说明和源码分析,帮助读者更好地理解Kafka服务端的工作原理。 ... [详细]
  • 预备知识可参考我整理的博客Windows编程之线程:https:www.cnblogs.comZhuSenlinp16662075.htmlWindows编程之线程同步:https ... [详细]
  • HashMap的相关问题及其底层数据结构和操作流程
    本文介绍了关于HashMap的相关问题,包括其底层数据结构、JDK1.7和JDK1.8的差异、红黑树的使用、扩容和树化的条件、退化为链表的情况、索引的计算方法、hashcode和hash()方法的作用、数组容量的选择、Put方法的流程以及并发问题下的操作。文章还提到了扩容死链和数据错乱的问题,并探讨了key的设计要求。对于对Java面试中的HashMap问题感兴趣的读者,本文将为您提供一些有用的技术和经验。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • Tomcat/Jetty为何选择扩展线程池而不是使用JDK原生线程池?
    本文探讨了Tomcat和Jetty选择扩展线程池而不是使用JDK原生线程池的原因。通过比较IO密集型任务和CPU密集型任务的特点,解释了为何Tomcat和Jetty需要扩展线程池来提高并发度和任务处理速度。同时,介绍了JDK原生线程池的工作流程。 ... [详细]
  • 集合的遍历方式及其局限性
    本文介绍了Java中集合的遍历方式,重点介绍了for-each语句的用法和优势。同时指出了for-each语句无法引用数组或集合的索引的局限性。通过示例代码展示了for-each语句的使用方法,并提供了改写为for语句版本的方法。 ... [详细]
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • 本文介绍了在Android开发中使用软引用和弱引用的应用。如果一个对象只具有软引用,那么只有在内存不够的情况下才会被回收,可以用来实现内存敏感的高速缓存;而如果一个对象只具有弱引用,不管内存是否足够,都会被垃圾回收器回收。软引用和弱引用还可以与引用队列联合使用,当被引用的对象被回收时,会将引用加入到关联的引用队列中。软引用和弱引用的根本区别在于生命周期的长短,弱引用的对象可能随时被回收,而软引用的对象只有在内存不够时才会被回收。 ... [详细]
author-avatar
lluuaalulua619
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有