核心逻辑概述
ThreadPoolExecutor是Java线程池中最核心的类之一,它能够保证线程池按照正常的业务逻辑执行任务,并通过原子方式更新线程池每个阶段的状态。
ThreadPoolExecutor类中存在一个workers工作线程集合,用户可以向线程池中添加需要执行的任务,workers集合中的工作线程可以直接执行任务,或者从任务队列中获取任务后执行。ThreadPoolExecutor类中提供了整个线程池从创建到执行任务,再到消亡的整个流程方法。本文,就结合ThreadPoolExecutor类的源码深度分析线程池执行任务的整体流程。
在ThreadPoolExecutor类中,线程池的逻辑主要体现在execute(Runnable)方法,addWorker(Runnable, boolean)方法,addWorkerFailed(Worker)方法和拒绝策略上,接下来,我们就深入分析这几个核心方法。
execute(Runnable)方法
execute(Runnable)方法的作用是提交Runnable类型的任务到线程池中。我们先看下execute(Runnable)方法的源码,如下所示。
public void execute(Runnable command) {
//如果提交的任务为空,则抛出空指针异常 if (command == null) throw new NullPointerException(); //获取线程池的状态和线程池中线程的数量 int c = ctl.get(); //线程池中的线程数量小于corePoolSize的值 if (workerCountOf(c) < corePoolSize) {
//重新开启线程执行任务 if (addWorker(command, true)) return; c = ctl.get(); } //如果线程池处于RUNNING状态,则将任务添加到阻塞队列中 if (isRunning(c) && workQueue.offer(command)) {
//再次获取线程池的状态和线程池中线程的数量,用于二次检查 int recheck = ctl.get(); //如果线程池没有未处于RUNNING状态,从队列中删除任务 if (! isRunning(recheck) && remove(command)) //执行拒绝策略 reject(command); //如果线程池为空,则向线程池中添加一个线程 else if (workerCountOf(recheck) == 0) addWorker(null, false); } //任务队列已满,则新增worker线程,如果新增线程失败,则执行拒绝策略 else if (!addWorker(command, false)) reject(command);}
整个任务的执行流程,我们可以简化成下图所示。
接下来,我们拆解execute(Runnable)方法,具体分析execute(Runnable)方法的执行逻辑。
(1)线程池中的线程数是否小于corePoolSize核心线程数,如果小于corePoolSize核心线程数,则向workers工作线程集合中添加一个核心线程执行任务。代码如下所示。
//线程池中的线程数量小于corePoolSize的值if (workerCountOf(c) < corePoolSize) {
//重新开启线程执行任务 if (addWorker(command, true)) return; c = ctl.get();}
(2)如果线程池中的线程数量大于corePoolSize核心线程数,则判断当前线程池是否处于R