Java线程池使用与原理

news/2024/9/16 16:04:12

线程池是什么?

我们可以利用java很容易创建一个新线程,同时操作系统创建一个线程也是一笔不小的开销。所以基于线程的复用,就提出了线程池的概念,我们使用线程池创建出若干个线程,执行完一个任务后,该线程会存在一段时间(用户可以设定空闲线程的存活时间,后面会介绍),等到新任务来的时候就直接复用这个空闲线程,这样就省去了创建、销毁线程损耗。当然空闲线程也会是一种资源的浪费(所有才有空闲线程存活时间的限制),但总比频繁的创建销毁线程好太多。

下面是我的测试代码

clipboard.png

clipboard.png

使用线程池5w线程运行完大约为400ms,不使用线程池运行大约为4350ms左右,其效率可见一斑(读者可以自行测试,不过由于电脑配置不一样,跑出来的数据会有差别,但使用线程池绝对是比创建线程要快的)。

java如何使用线程池?

上面的测试代码中已经使用了线程池,下面正式介绍一下。

java所有的线程池最顶层是一个Executor接口,其只有一个execute方法,用于执行所有的任务,java又提供了ExecutorService接口继承自Executor并且扩充了一下方法,在往下就是AbstractExecutorService这个抽象类,其实现了ExecutorService,最后就是ThreadPoolExecutor其继承自上面的抽象类,我们常使用的java线程池就是创建的这个类的实例。

而上面我们使用Executors是一个工具类,它就是一个语法糖,为我们把各种不同的业务的线程池参数进行封装,进行new操作。

clipboard.png

上面就是Executors.newFixedThreadPool(10)的源码。

下面重点来了,说一说ThreadPoolExecutor构造方法各参数的意思。

clipboard.png

上面这个构造方法是最全的。

下面我们根据源码来解释部分参数意思,这样更有说服力。

下面是ThreadPoolExecutor#execute方法,就是我们上面接口调用的execute实际执行者。

clipboard.png

ctl是一个AtomicInteger实例,是一个提供了原子语句的CAS操作的类,它用来记录线程池中当前运行的线程数量加上-2^29,workCountOf方法就取得其绝对值(可以去看源码如何实现),当其小于corePoolSize时,会调用addWorker方法(是用来创建一个新Workder,Workder会创建一个Thread,所以就是创建线程的方法),addWorkd创建线程过程中会跟corePoolSize或者maxnumPoolSize的值比较(当传入true会根corePoolSize比较,false会根据maxnumPoolSize比较,大于等于其值会创建失败)。可见如何当前运行中的线程数量小于corePoolSize就是创建并且也会创建成功(
只简单的讨论线程池Running状态下)。

如果当运行中线程数大于等于corePoolSize时,进入第二个if,isRunning是跟SHUTDOWN(其值=0)比较,之前说过c等于当前运行的线程数量加上-2^29,如果当前当前运行的线程数据达到2^29时其值就=0,isRunning返回false,else中在执行addWorkd也会返回false(addWorkd也对其进行了检验),所以这表示线程池最多能支持2^29个线程同时运行(足够用了)。

workQueue.offer(command)就是将runnable加入等待队列,加入等待队列后runWorker方法会从队列中获取任务执行的。如果当前队列采用的是有界队列(ArrayBlockingQueue)当队列满了offer就会返回false,这是就进入else if,看!这里传入了false,说明这里要跟maxnumPoolSize比较了,如果这里运行的线程数大于等于maxnumPoolSize,那么这个线程任务就要被线程池拒绝了,执行reject(command),拒绝方法中使用了我们ThreadPoolExecutor构造方法中的RejectedExecutionHandler(拒绝策略),后面再详细解释。

经过上面的结合源码的介绍,下面对们ThreadPoolExecutor的参数介绍就好理解了。

在此我向大家推荐一个架构学习交流群。交流学习群号:575745314 里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

线程池中线程创建和拒绝策略

corePoolSize,maxnumPoolSize,BlockingQueue这三个要一块说

当线程池运行的线程小于corePoolSize时,来一个新线程任务总是会新建一个线程来执行;当大于corePoolSize就会把任务加入到等待队列blockingQueue中,如果你传入的BlockingQueue是一个无界队列(LinkedBlockingQueue)这是队列可以存放“无穷多”的任务,所有总是会加入队列成功,跟maxnumPoolSize就没关系了,这也表示线程池中线程数最多为corePoolSize个;但是如果你传入的是有界队列(ArrayBlockingQueue,SynchronousQueue),当队列满时,并且线程数小于maxmunPoolSize就是创建新的线程直至线程数大于maxnumPoolSize;如果当线程数量大于maxnumPoolSize时,在加入任务就会被线程池拒绝。

RejectedExecutionHandler拒绝策略java给实现了4个AbortPolicy,CallerRunsPolicy,DiscardOldestPolicy,DiscardPolicy用户也可以自己实现该接口实现自己的拒绝策略;第一个就是直接抛出异常,我们可以进行trycatch处理;第二个就是该新任务直接运行;第三个是取消队列中最老的;第四个是取消当前任务。


http://lihuaxi.xjx100.cn/news/272792.html

相关文章

配置telnet

配置telnet<?xml:namespace prefix o ns "urn:schemas-microsoft-com:office:office" />允许root账号能够登录telnet&#xff0c;但是拒绝某一台主机登录且只允许在9&#xff1a;00-14&#xff1a;00 14&#xff1a;00-18&#xff1a;00能够访问&#xff0…

使用 OpenCV 和 Tesseract 对图像中的感兴趣区域 (ROI) 进行 OCR

点击上方“小白学视觉”&#xff0c;选择加"星标"或“置顶”重磅干货&#xff0c;第一时间送达在这篇文章中&#xff0c;我们将使用 OpenCV 在图像的选定区域上应用 OCR。在本篇文章结束时&#xff0c;我们将能够对输入图像应用自动方向校正、选择感兴趣的区域并将OC…

【SICP练习】110 练习3.23

练习3-23 原文 Exercise 3.23. A deque (“double-ended queue”) is a sequence in which items can be inserted and deleted at either the front or the rear. Operations on deques are the constructor make-deque, the predicate empty-deque?, selectors front-deque …

外国教授在B站当UP主上课,网友直呼好家伙:滑铁卢大学《差分隐私》课程上线...

视学算法报道作者&#xff1a;泽南、蛋酱Gautam Kamath 的课程&#xff0c;让网友们直呼好家伙。一位加拿大高校的教授&#xff0c;不远万里来到 B 站&#xff0c;一言不合就扔下了 24 个小时的课程视频&#xff0c;这是一种什么样的精神&#xff1f;1 月 3 日&#xff0c;滑铁…

用 float 存储金额,老板说损失从工资里扣!

点击上方“方志朋”&#xff0c;选择“设为星标”回复”666“获取新整理的面试文章本文来源&#xff1a;juejin.im/post/5c08db5ff265da611e4d7417《Java工程师面试突击&#xff08;第3季&#xff09;》重磅升级&#xff0c;由原来的70讲增至160讲&#xff0c;内容扩充一倍多&a…

先马后看!详解线性回归、朴素贝叶斯、随机森林在R和Python中的实现应用!(附代码)...

来源| analyticsvidhya编译| 火火酱&#xff0c;责编| Carol出品 | AI科技大本营&#xff08;ID&#xff1a;rgznai100&#xff09;谷歌的自动驾驶汽车和机器人得到了媒体的广泛关注&#xff0c;但是公司真正的未来是在机器学习领域&#xff0c;这一技术会使计算机变得更加智能…

V神再为BCH发声!

近日&#xff0c;以太坊创始人Vitalik Buterin在回应一篇质疑以太坊发展路线的文章中&#xff0c;在回应对方所提出的问题后&#xff0c;在文末为BCH说了两句话。 翻译&#xff1a;你使用的术语“BCash”是错误的&#xff0c;它是一个现实品&#xff0c;不是一个区块链/加密货币…

基于自适应显着性的图像分割(源码开放)

点击上方“小白学视觉”&#xff0c;选择加"星标"或“置顶”重磅干货&#xff0c;第一时间送达本文介绍算法的源码在github上给出https://github.com/TimChinenov/GraspPicture前言成产品及系统平台的现场演示&#xff0c;编写技术应用服务方案等&#xff0c;编写投标…