Java线程池、拒绝策略

169 阅读1分钟

ThreadPoolExecutor构造器

/**
     * The default rejected execution handler
     */
private static final RejectedExecutionHandler defaultHandler = new AbortPolicy();

public ThreadPoolExecutor(int corePoolSize,
                          int maximumPoolSize,
                          long keepAliveTime,
                          TimeUnit unit,
                          BlockingQueue<Runnable> workQueue) {
    this(corePoolSize, maximumPoolSize, keepAliveTime, unit, workQueue,
         Executors.defaultThreadFactory(), defaultHandler);
}
参数含义备注
corePoolSize核心线程数量,线程池初始化时定义如果corePoolSize==maximumPoolSize,则线程池中线程不会空闲,keepAliveTime就不再有作用。
maximumPoolSize最大线程数;核心线程与非核心线程共同使用线程池,但核心线程不会被回收。
keepAliveTime保持时间:若目前线程数>核心线程数,多余线程在等待keepAliveTime时间后未进行任务,则会被回收。
unitkeepAliveTime的时间单位
workQueue等待队列
defaultHandler拒绝策略AbortPolicy(默认):直接抛弃

异常产生原因

当线程池里的线程都繁忙的时候,新任务会被提交给阻塞队列保存,当提交给阻塞队列的任务,超出了该队列的最大容量时。线程池就会拒绝接收新任务,随即抛出异常。

四种拒绝策略

策略名称含义备注
AbortPolicy当队列满时抛出RejectExecutionException异常,可捕捉后自行处理默认策略
通过异常可及时发现系统无法承载的并发
DiscardPolicy默默丢弃任务,不抛出异常无法发现系统的异常状态,且影响业务(对于阅读量统计的低重要性的场景可使用)
DiscardOldestPolicy丢弃队列中最前的任务,重新提交被拒绝的任务“喜新厌旧”策略
CallerRunsPolicy由调用线程(当前提交任务的线程)处理该任务不抛出异常,任务仍然执行