前几天,技术群里有个群友问了一个关于线程池的问题,内容如图所示:

那么就来和大家探讨下这个问题,在线程池中,线程会从 workQueue 中读取任务来执行,最小的执行单位就是 Worker,Worker 实现了 Runnable 接口,重写了 run 方法,这个 run 方法是让每个线程去执行一个循环,在这个循环代码中,去判断是否有任务待执行,若有则直接去执行这个任务,因此线程数不会增加。

如下是线程池创建线程的整体流程图:

首先会判断线程池的状态,也就是是否在运行,若线程为非运行状态,则会拒绝。接下来会判断线程数是否小于核心线程数,若小于核心线程数,会新建工作线程并执行任务,随着任务的增多,线程数会慢慢增加至核心线程数,如果此时还有任务提交,就会判断阻塞队列 workQueue 是否已满,若没满,则会将任务放入到阻塞队列中,等待工作线程获得并执行,如果任务提交非常多,使得阻塞队达到上限,会去判断线程数是否小于最大线程数 maximumPoolSize,若小于最大线程数,线程池会添加工作线程并执行任务,如果仍然有大量任务提交,使得线程数等于最大线程数,如果此时还有任务提交,就会被拒绝。

现在我们对这个流程大致有所了解,那么让我们去看看源码是如何实现的吧!

线程池的任务提交从 submit 方法来说,submit 方法是 AbstractExecutorService 抽象类定义的,主要做了两件事情:

把 Runnable 和 Callable 都转化成 FutureTask

使用 execute 方法执行 FutureTask

execute 方法是 ThreadPoolExecutor 中的方法,源码如下:

public void execute(Runnable command) {

// 若任务为空,则抛 NPE,不能执行空任务

if (command == null) {

throw new NullPointerException();

}

int c = ctl.get();

// 若工作线程数小于核心线程数,则创建新的线程,并把当前任务 command 作为这个线程的第一个任务

if (workerCountOf(c) < corePoolSize) {

if (addWorker(command, true)) {

return;

}

c = ctl.get();

}

/**

* 至此,有以下两种情况:

* 1.当前工作线程数大于等于核心线程数

* 2.新建线程失败

* 此时会尝试将任务添加到阻塞队列 workQueue

*/

// 若线程池处于 RUNNING 状态,将任务添加到阻塞队列 workQueue 中

if (isRunning(c) && workQueue.offer(command)) {

// 再次检查线程池标记

int recheck = ctl.get();

// 如果线程池已不处于 RUNNING 状态,那么移除已入队的任务,并且执行拒绝策略

if (!isRunning(recheck) && remove(command)) {

// 任务添加到阻塞队列失败,执行拒绝策略

reject(command);

}

// 如果线程池还是 RUNNING 的,并且线程数为 0,那么开启新的线程

else if (workerCountOf(recheck) == 0) {

addWorker(null, false);

}

}

/**

* 至此,有以下两种情况:

* 1.线程池处于非运行状态,线程池不再接受新的线程

* 2.线程处于运行状态,但是阻塞队列已满,无法加入到阻塞队列

* 此时会尝试以最大线程数为界创建新的工作线程

*/

else if (!addWorker(command, false)) {

// 任务进入线程池失败,执行拒绝策略

reject(command);

}

}

可以看到 execute 方法中的的核心方法为 addWorker,再去看 addWorker 方法之前,先看下 Worker 的初始化方法:

Worker(Runnable firstTask) {

// 每个任务的锁状态初始化为-1,这样工作线程在运行之前禁止中断

setState(-1);

this.firstTask = firstTask;

// 把 Worker 作为 thread 运行的任务

this.thread = getThreadFactory().newThread(this);

}

在 Worker 初始化时把当前 Worker 作为线程的构造器入参,接下来从 addWorker 方法中可以找到如下代码:

final Thread t = w.thread;

// 如果成功添加了 Worker,就可以启动 Worker 了

if (workerAdded) {

t.start();

workerStarted = true;

}

这块代码是添加 worker 成功,调用 start 方法启动线程,Thread t = w.thread; 此时的 w 是 Worker 的引用,那么t.start();实际上执行的就是 Worker 的 run 方法。

Worker 的 run 方法中调用了 runWorker 方法,简化后的 runWorker 源码如下:

final void runWorker(Worker w) {

Runnable task = w.firstTask;

while (task != null || (task = getTask()) != null) {

try {

task.run();

} finally {

task = null;

}

}

}

这个 while 循环有个 getTask 方法,getTask 的主要作用是阻塞从队列中拿任务出来,如果队列中有任务,那么就可以拿出来执行,如果队列中没有任务,这个线程会一直阻塞到有任务为止(或者超时阻塞),其中 getTask 方法的时序图如下:

其中线程复用的关键是 1.6 和 1.7 部分,这部分源码如下:

Runnable r = timed ? workQueue.poll(keepAliveTime, TimeUnit.NANOSECONDS) : workQueue.take();

使用队列的 poll 或 take 方法从队列中拿数据,根据队列的特性,队列中有任务可以返回,队列中无任务会阻塞。

线程池的线程复用就是通过取 Worker 的 firstTask 或者通过 getTask 方法从 workQueue 中不停地取任务,并直接调用 Runnable 的 run 方法来执行任务,这样就保证了每个线程都始终在一个循环中,反复获取任务,然后执行任务,从而实现了线程的复用。

总结

本文主要从源码的角度解析了 Java 线程池中的线程复用是如何实现的。欢迎大家留言交流讨论。

最好的关系就是互相成就,大家的在看、转发、留言三连就是我创作的最大动力。

更详细的源码解析可以点击链接查看:https://github.com/wupeixuan/JDKSourceCode1.8

参考

https://github.com/wupeixuan/JDKSourceCode1.8

面试官系统精讲Java源码及大厂真题

Java并发编程学习宝典

Java 并发面试 78 讲

java线程池newfi_Java 线程池中的线程复用是如何实现的?相关推荐

  1. java线程池怎么创建_java中的线程池,如何创建?

    Java中的线程池它是线程的容器,或者(换句话说,它是具有执行任务能力的线程的集合). 我们可以使用ThreadPool框架来定位(或实现)线程池. 线程池可以包含多个线程.每当我们执行任何任务时,线 ...

  2. android 线程池 怎么用,android中的线程池 怎么用

    满意答案 qpierq0n5 2016.05.02 采纳率:52%    等级:13 已帮助:8164人 //在Android中实现线程池,首先需要实现一个线程工厂(ThreadFactory)的子类 ...

  3. c语言数据库线程池,C语言多线程中运行线程池,在线程池中运行线程池,,传递的结构体参数值为空/NULL/0...

    typedef struct { }LoanInfos; typedef struct{ int cp;//主线程编号 int thread;//线程编号 long int time; int arr ...

  4. Java核心(三)并发中的线程同步与锁

    2019独角兽企业重金招聘Python工程师标准>>> 乐观锁.悲观锁.公平锁.自旋锁.偏向锁.轻量级锁.重量级锁.锁膨胀...难理解?不存的!来,话不多说,带你飙车. 上一篇介绍了 ...

  5. java中线程的生命周期_Java中的线程生命周期– Java中的线程状态

    java中线程的生命周期 Understanding Thread Life Cycle in Java and Thread States are very important when you a ...

  6. java异步线程数_spring异步service中处理线程数限制详解

    情况简介 spring项目,controller异步调用service的方法,产生大量并发. 具体业务: 前台同时传入大量待翻译的单词,后台业务接收单词,并调用百度翻译接口翻译接收单词并将翻译结果保存 ...

  7. 如何查找历史线程阻塞原因_java并发编程-线程状态,线程阻塞方式,阻塞中的线程如何终止?...

    前面的例子通过volatile boolean来让任务终止,那么如果任务阻塞了? 如何终止它?本文来看下,参考think in java . 线程的状态 1.new:线程创建后的短暂状态,其分配系统资 ...

  8. C# 线程手册 第二章 .NET 中的线程 线程的优势

    额,我猜你现在可能会这么想"既然线程会对我的程序产生负面影响,那么我为什么要使用它呢?".其实问题的关键不在于到底用不用线程,而在于何时何地使用线程.知道在什么情况下应该使用线程是 ...

  9. python进程线程协程区别_Python中 进程 线程 协程

    一.进程 进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位,是操作系统结构的基础.在早期面向进程设计的计算机结构中,进程是程序的基本执行实体:在 ...

  10. python进程线程处理模块_python程序中的线程操作 concurrent模块使用详解

    一.concurrent模块的介绍 concurrent.futures模块提供了高度封装的异步调用接口 ThreadPoolExecutor:线程池,提供异步调用 ProcessPoolExecut ...

最新文章

  1. Software development Problem
  2. 002——php字符串中的处理函数(一)
  3. 十一、加权线性回归案例:预测鲍鱼的年龄
  4. 柱状图、堆叠柱状图、瀑布图有什么区别?怎样用Python绘制?
  5. 【Spring Bean的生命周期】
  6. 龟兔赛跑预测(蓝桥杯)
  7. AutoLISP圆内接多边形
  8. 百度App网络深度优化系列(一):DNS优化
  9. Atitit 遍历文件夹算法 autoit attilax总结
  10. 华硕FX50JK4200安装Win8.1后如何禁用触摸板
  11. Visual Studio2010打开界面文件时报错:“未在此计算机上注册activex控件{648A5600-2C6E-101B-82B6-000000000014}”
  12. PostgresSQL弱密码导致命令执行
  13. WebService系列之HttpClient调用WebService接口
  14. 鼠标放上去,变成小手状
  15. ubuntu 如何放大终端窗口字体
  16. python数据分析与可视化从入门到精通_零基础学Python爬虫、数据分析与可视化从入门到精通...
  17. 娱乐至死,年轻人的慢性毒药
  18. Arduino IDE环境下WeMos D1开发板引脚定义和映射
  19. 脚本---perl与python的比较
  20. 甲骨文大中华区人事变动:卢汝文退休

热门文章

  1. Vivado安装找不到matlab,vivado安装System Generator不支持新版Matlab怎么办?
  2. tkinter-canvas详解
  3. 死磕ppt--图片处理
  4. C语言基础:求解一元二次方程
  5. python用什么软件编程-初学 Python 需要安装哪些软件?
  6. 提高网络营销的转化只需掌握这四步
  7. 机器学习数据挖掘笔记_18(PGM练习二:贝叶斯网络在遗传图谱在的应用)
  8. 微信小程序在组件中刷新当前页面
  9. Python爬虫数据存储不同格式在excel表中通用代码
  10. Moon Player正式登陆爱奇艺-奇遇VR应用商店