四种线程池拒绝策略

四种线程池拒绝策略一 前言线程池 相信很多人都有用过 没用过相信的也有学习过 但是 线程池的拒绝策略 相信知道的人会少许多 二 四种线程池拒绝策略当线程池的任务缓存队列已满并且线程池中的线程数目达到 maximumPoolS 时 如果还有任务到来就会采取任务拒绝策略 通常有以下四种策略 ThreadPoolEx AbortPolicy 丢弃任务并抛出 RejectedExec

一、前言

二、四种线程池拒绝策略

三、线程池默认的拒绝策略

java.util.concurrent.ThreadPoolExecutor类的源码,我们可以看到:

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

线程池的默认拒绝策略为AbortPolicy,即丢弃任务并抛出RejectedExecutionException异常。我们可以通过代码来验证这一点,现有如下代码:

public class ThreadPoolTest {

public static void main(String[] args) {

BlockingQueue

queue = new ArrayBlockingQueue<>(100);

ThreadFactory factory = r -> new Thread(r, “test-thread-pool”);
ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 5,
0L, TimeUnit.SECONDS, queue, factory);
while (true) {

executor.submit(() -> {

try {

System.out.println(queue.size());
Thread.sleep(10000);
} catch (InterruptedException e) {

e.printStackTrace();
}
});
}
}
}



































这里是一个默认的线程池,没有设置拒绝策略,设置了最大线程队列是100。运行代码:

四种线程池拒绝策略

四、设置线程池拒绝策略

四种线程池拒绝策略

四种线程池拒绝策略


* 配置类实现AsyncConfigurer接口并重写getAsyncExcutor方法,并返回一个ThreadPoolTaskExevutor
* 这样我们就获得了一个基于线程池的TaskExecutor
*/









@Override
public Executor getAsyncExecutor() {

ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
taskExecutor.setCorePoolSize(CORE_POOL_SIZE);
taskExecutor.setMaxPoolSize(MAX_POOL_SIZE);
taskExecutor.setQueueCapacity(QUEUE_CAPACITY);
taskExecutor.initialize();
taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.AbortPolicy());
return taskExecutor;
}
}

五、拒绝策略场景分析

ThreadPoolExecutor.AbortPolicy:丢弃任务并抛出RejectedExecutionException异常。

A handler for rejected tasks that throws a {@code RejectedExecutionException}.

这是线程池默认的拒绝策略,在任务不能再提交的时候,抛出异常,及时反馈程序运行状态。如果是比较关键的业务,推荐使用此拒绝策略,这样子在系统不能承载更大的并发量的时候,能够及时的通过异常发现。

(2)DiscardPolicy

ThreadPoolExecutor.DiscardPolicy:丢弃任务,但是不抛出异常。如果线程队列已满,则后续提交的任务都会被丢弃,且是静默丢弃。

A handler for rejected tasks that silently discards therejected task. 

使用此策略,可能会使我们无法发现系统的异常状态。建议是一些无关紧要的业务采用此策略。例如,本人的博客网站统计阅读量就是采用的这种拒绝策略。

(3)DiscardOldestPolicy

ThreadPoolExecutor.DiscardOldestPolicy:丢弃队列最前面的任务,然后重新提交被拒绝的任务。

A handler for rejected tasks that discards the oldest unhandled request and then retries {@code execute}, unless the executor is shut down, in which case the task is discarded.

此拒绝策略,是一种喜新厌旧的拒绝策略。是否要采用此种拒绝策略,还得根据实际业务是否允许丢弃老任务来认真衡量。

(4)CallerRunsPolicy

ThreadPoolExecutor.CallerRunsPolicy:由调用线程处理该任务

A handler for rejected tasks that runs the rejected task directly in the calling thread of the {@code execute} method, unless the executor has been shut down, in which case the task is discarded. 

如果任务被拒绝了,则由调用线程(提交任务的线程)直接执行此任务,我们可以通过代码来验证这一点:

把之前的代码修改如下:

public static void main(String[] args) { BlockingQueue 
                                 
                                   queue = new ArrayBlockingQueue<>(10); ThreadFactory factory = r -> new Thread(r, "test-thread-pool"); ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 5, 0L, TimeUnit.SECONDS, queue, factory, new ThreadPoolExecutor.CallerRunsPolicy()); for (int i = 0; i < 1000; i++) { executor.submit(() -> { try { System.out.println(Thread.currentThread().getName() + ":执行任务"); Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } }); } } 
                                 

 

四种线程池拒绝策略

 

把队列最大值改为10,打印输出线程的名称。执行结果如下:

 

四种线程池拒绝策略

 

通过结果可以看到,主线程main也执行了任务,这正说明了此拒绝策略由调用线程(提交任务的线程)直接执行被丢弃的任务的。

六、总结
































版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/199541.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月20日 下午12:46
下一篇 2026年3月20日 下午12:46


相关推荐

  • 腾讯元宝强势崛起,跻身应用TOP3!

    腾讯元宝强势崛起,跻身应用TOP3!

    2026年3月13日
    2
  • Dubbo框架介绍「建议收藏」

    Dubbo是一个常用的分布式服务框架,它致力于提供高性能和透明化的RPC远程调用服务方案,Dubbo有助于开发企业级的开发效率,以及可以通过简单的配置就可以做到负载均衡。   一、Dubbo的基础知识   1.Dubbo是什么   2.Dubbo涉及的知识      二、Dubbo框架设计介绍   1.Dubbo的各个角色

    2022年4月13日
    34
  • java微信接口开发源代码_基于springMVC的java微信公众平台开发源码

    java微信接口开发源代码_基于springMVC的java微信公众平台开发源码packagecom weixin robot importjava io BufferedRead importjava io IOException importjava io InputStreamR importjava io UnsupportedE import 链接已屏蔽 HttpURLConne import 链接已

    2026年3月16日
    3
  • 增量表全量表拉链表区别_hive 增量数据更新

    增量表全量表拉链表区别_hive 增量数据更新一、概念增量表:记录更新周期内新增的数据,即在原表中数据的基础上新增本周期内产生的新数据;全量表:记录更新周期内的全量数据,无论数据是否有变化都需要记录;拉链表:一种数据存储和处理的技术方式,可以记录数据的历史信息,记录数据从开始一直到当前所有变化的信息。二、举例详解增量表:以页面访问数据表为例,假设该表从2020-06-01开始记录数据,按天更新,分区为dt。2020-06-01产生了三条访问数据,如下表:2020-06-02首页和商详页又产生了2条访问数据,该两条即为2020-06-

    2022年10月17日
    4
  • 常用的Js调Android方法,以及Android原生传值给Js

    常用的Js调Android方法,以及Android原生传值给Js

    2021年3月12日
    175
  • 深入理解B/S与C/S架构

    深入理解B/S与C/S架构深入理解 B S 与 C S 架构阅读目录 什么是 C S 架构 什么是 B S 架构 B S 架构的几种形式 发展前景 一 什么是 C S 架构 C S 架构是第一种比较早的软件架构 主要用于局域网内 也叫客户机 服务器模式 它可以分为客户机和服务器两层 第一层 在客户机系统上结合了界面显示与业务逻辑 第二层 通过网络结合了数据库服务器

    2026年3月16日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号