在线客服
扫描二维码
下载博学谷APP
扫描二维码
关注博学谷微信公众号
ThreadPoolExecutor里面使用到JUC同步器框架AbstractQueuedSynchronizer、大量的位操作、CAS操作。ThreadPoolExecutor提供了固定活跃线程、额外的线程、任务队列以及拒绝策略这几个重要的功能。下面我们一起来看看Java 线程池ThreadPoolExecutor的原理解析。

1、JUC同步器框架
ThreadPoolExecutor里面使用到JUC同步器框架,主要用于四个方面:
(1)全局锁mainLock成员属性,是可重入锁ReentrantLock类型,主要是用于访问工作线程Worker集合和进行数据统计记录时候的加锁操作。
(2)条件变量termination,Condition类型,主要用于线程进行等待终结awaitTermination()方法时的带期限阻塞。
(3)任务队列workQueue,BlockingQueue类型,任务队列,用于存放待执行的任务。
(4)工作线程,内部类Worker类型,是线程池中真正的工作线程对象。
2、核心线程
这里先参考ThreadPoolExecutor的实现并且进行简化,实现一个只有核心线程的线程池,要求如下:暂时不考虑任务执行异常情况下的处理;任务队列为无界队列;线程池容量固定为核心线程数量;暂时不考虑拒绝策略。
public class CoreThreadPool implements Executor {
private BlockingQueue<Runnable> workQueue;
private static final AtomicInteger COUNTER = new AtomicInteger();
private int coreSize;
private int threadCount = 0;
public CoreThreadPool(int coreSize) {
this.coreSize = coreSize;
this.workQueue = new LinkedBlockingQueue<>();
}
@Override
public void execute(Runnable command) {
if (++threadCount <= coreSize) {
new Worker(command).start();
} else {
try {
workQueue.put(command);
} catch (InterruptedException e) {
throw new IllegalStateException(e);
}
}
}
private class Worker extends Thread {
private Runnable firstTask;
public Worker(Runnable runnable) {
super(String.format("Worker-%d", COUNTER.getAndIncrement()));
this.firstTask = runnable;
}
@Override
public void run() {
Runnable task = this.firstTask;
while (null != task || null != (task = getTask())) {
try {
task.run();
} finally {
task = null;
}
}
}
}
private Runnable getTask() {
try {
return workQueue.take();
} catch (InterruptedException e) {
throw new IllegalStateException(e);
}
}
public static void main(String[] args) throws Exception {
CoreThreadPool pool = new CoreThreadPool(5);
IntStream.range(0, 10)
.forEach(i -> pool.execute(() ->
System.out.println(String.format("Thread:%s,value:%d", Thread.currentThread().getName(), i))));
Thread.sleep(Integer.MAX_VALUE);
}
}
某次运行结果如下:
Thread:Worker-0,value:0
Thread:Worker-3,value:3
Thread:Worker-2,value:2
Thread:Worker-1,value:1
Thread:Worker-4,value:4
Thread:Worker-1,value:5
Thread:Worker-2,value:8
Thread:Worker-4,value:7
Thread:Worker-0,value:6
Thread:Worker-3,value:9
设计此线程池的时候,核心线程是懒创建的,如果线程空闲的时候则阻塞在任务队列的take()方法,其实对于ThreadPoolExecutor也是类似这样实现,只是如果使用了keepAliveTime并且允许核心线程超时则会使用BlockingQueue#poll进行轮询代替永久阻塞。
3、其他附加功能
构建ThreadPoolExecutor实例的时候,需要定义maximumPoolSize(线程池最大线程数)和corePoolSize(核心线程数)。当任务队列是有界的阻塞队列,核心线程满负载,任务队列已经满的情况下,会尝试创建额外的maximumPoolSize - corePoolSize个线程去执行新提交的任务。当ThreadPoolExecutor这里实现的两个主要附加功能是:
(1)一定条件下会创建非核心线程去执行任务,非核心线程的回收周期(线程生命周期终结时刻)是keepAliveTime,线程生命周期终结的条件是:下一次通过任务队列获取任务的时候并且存活时间超过keepAliveTime。
(2)提供拒绝策略,也就是在核心线程满负载、任务队列已满、非核心线程满负载的条件下会触发拒绝策略。
以上就是Java 线程池ThreadPoolExecutor的原理解析,大家都看懂了吗?如果想要深入学习更多关于Java 线程池的内容,欢迎大家在博学谷报名在线课程进行学习~
— 申请免费试学名额 —
在职想转行提升,担心学不会?根据个人情况规划学习路线,闯关式自适应学习模式保证学习效果
讲师一对一辅导,在线答疑解惑,指导就业!
相关推荐 更多
自学Java开发要多久?要学到什么程度才能就业?
自学Java开发要多久?要学到什么程度才能就业?自学能力强的话,完全熟练掌握Java开发技能最快也需要半年,必须要学的内容包括主流技术框架、 Lambda表达式及其他新特性、Spring等现阶段企业流行技术和知识点。下面小编详细讲一讲Java开发就业的标准。
6579
2019-09-19 10:26:23
Java开发者应该如何提升自己?Java学习路线规划
对于Java的开发者来说,想要在工作中不断提升自己,需要一直保持学习的态度。那么在学习的道路上,哪些技术和工具框架需要Java开发者掌握呢?在这里,我将和大家一起探讨,那些Java开发者必须学习和掌握的工具、程序库、框架和API。
5392
2019-11-09 20:09:08
Java培训班在线教育怎么样?零基础能学好吗?
Java培训班在线教育怎么样?零基础能学好吗?随着IT在线教育的发展,博学谷的课程无论是在教学方式还是在课程内容上,都实现了极大的突破和升级,一改传统线上教育只对学员单向输出的不足。以博学谷Java在线就业班的课程为例,该教程专门为零基础学员打造,因此课程难度循序渐进,逐步深入,只要愿意付出努力好好学习,即使是零基础学完该课程也能进阶为Java高手。
4778
2020-01-06 15:26:07
黑马头条项目实战学什么?
随着智能手机的普及和生活节奏的加快,人们更加习惯于利用碎片时间通过手机来看新闻。 因此,现在对于移动资讯客户端的需求也越来越高,黑马头条项目正是在这样背景下研发出来的。 黑马头条项目是对在线教育平台业务进行大数据统计分析的系统,碎片化、切换频繁、社交化和个性化现如今成为人们阅读行为的标签。那么,黑马头条项目实战学什么呢?
7860
2020-04-28 18:37:32
学习Java技术开发被劝退的原因有哪些?
学习Java技术开发被劝退的原因有哪些?有不少人学习Java时会崩溃,如Java糟糕的入门体验、核心标准库不够用、最啰嗦的语言Java、复杂的 "企业级"产品、库作者处理的问题越来越复杂、太多的抽象性、非本地的FP支持等等原因。
3708
2022-04-12 11:36:12
