Bootstrap

hibernate 并发获取session失败 空指针_高并发之|通过ThreadPoolExecutor类的源码深度解析线程池执行任务的核心流程...

bbca484eb18af7cb6be8940331c43159.png84302f417ca3a8a031a69d54ba794031.png

核心逻辑概述

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);}

整个任务的执行流程,我们可以简化成下图所示。

969d071d653486ddecf5536fef95f31d.png

接下来,我们拆解execute(Runnable)方法,具体分析execute(Runnable)方法的执行逻辑。

(1)线程池中的线程数是否小于corePoolSize核心线程数,如果小于corePoolSize核心线程数,则向workers工作线程集合中添加一个核心线程执行任务。代码如下所示。

//线程池中的线程数量小于corePoolSize的值if (workerCountOf(c) < corePoolSize) {
      //重新开启线程执行任务  if (addWorker(command, true))    return;  c = ctl.get();}

(2)如果线程池中的线程数量大于corePoolSize核心线程数,则判断当前线程池是否处于R

;