当FutureTask遇上DiscardPolicy,有坑

news2025/1/19 20:17:53

文章目录

    • 有啥坑呢?
    • 知识回顾
    • 问题触发条件
    • 问题复现
    • 问题分析
    • 问题修复
    • 扩展

哈喽,你好,我是余数。今天来了解下当 FutureTask 遇上 DiscardPolicyDiscardOldestPolicy 时容易掉的坑,然后分析分析问题产生的原因以及如何规避这类问题。

有啥坑呢?

获取异步执行结果时可能会导致主线程阻塞。
假如我向线程池中提交了5个带返回值的任务,编号为从0到4,但是我可能只能够获取到任务0和任务1的结果,获取任务2时主线程阻塞。

知识回顾

分析问题之前,我们先回顾一下JDK线程池自带的4个拒绝策略,它们分别是:

  1. CallerRunsPolicy:使用调用方线程中执行被拒绝的任务。
  2. AbortPolicy:拒绝任务,并抛出RejectedExecutionException异常。ThreadPoolExecutorScheduledThreadPoolExecutor的默认处理程序。或者任意其他自定义的,直接抛弃任务,但不抛出异常的策略。
  3. DiscardPolicy:直接丢弃被拒绝的任务,不抛出任何异常。
  4. DiscardOldestPolicy:丢弃队列中最老的任务,不抛出任何异常。

关于FutureTask,我们主要用它来获取异步线程的执行结果。在异步多线程的使用中又必定离不开线程池,使用线程池就一定会用到线程池的拒绝策略,因为不管多大的池子,它总是会满的不是吗?

问题触发条件

什么时候会出现这个问题呢?

  1. 向线程池提交(submit)任务,返回 Future。或者让线程池直接执行(execute)带返回值的任务FutureTask
  2. 线程池满,触发拒绝策略。且拒绝策略是 DiscardPolicy 或者 DiscardOldestPolicy
  3. 在发起异步的主线程调用 FutureTaskget()方法获取异步执行结果。

问题复现

为了方便,我使用了Spring的ThreadPoolTaskExecitor,其底层实现还是JDK的ThreadPoolExecutor。核心线程数、最大线程数和队列容量全部设置为1,主要是为了快速触发线程池的拒绝策略。

import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.*;

public class TestFutureTaskAndDiscardPolicy {

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(1);
        executor.setMaxPoolSize(1);
        executor.setQueueCapacity(1);
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.DiscardPolicy());
        executor.initialize();

        // FutureTask<String> futureTask = new FutureTask<String>(new MyCallable("任务-" + i));
        // executor.execute(futureTask);

        List<Future> res = new ArrayList<>();
        for (int i = 0; i < 5; i++) {
            Future<String> futureTask = executor.submit(new MyCallable("任务-" + i));
            res.add(futureTask);
        }

        for (int i = 0; i < 5; i++) {
            Future<String> futureTask = res.get(i);
            System.out.println(futureTask.get());
        }

    }

    static class MyCallable implements Callable<String> {
        private String name;

        public MyCallable(String name) {
            this.name = name;
        }

        @Override
        public String call() throws Exception {
            System.out.println(name + ":开始。。。");
            try {
                Thread.sleep(1_000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            System.out.println(name + ":结束。。。");
            return name + ":成功返回";
        }
    }
}

当FutureTask遇上DiscardPolicy,有坑

问题分析

首先可以定位到问题是出现在 获取异步任务结果 这里,那么为什么这里会出现阻塞呢?
当FutureTask遇上DiscardPolicy,有坑

答案就在FutureTask的源码里,查看源码会发现,当FutureTask的状态(state)小于等于 COMPLETING 时,线程会等待,直到这个任务完成或者等待超时。因为默认等待时间是0,也就是不会超时,线程会一直等待下去,也就是我们文章开始提到的坑,主线程卡死了。
当FutureTask遇上DiscardPolicy,有坑
当FutureTask遇上DiscardPolicy,有坑

现在我们知道了阻塞的原因是因为 FutureTask 的状态不对,那么 FutureTask 的状态又是为啥不对呢?
我们继续从源码找起,当我们向线程池提交一个带返回值任务的时候,线程池会将任务封装成一个FutureTask对象,而FutureTask对象默认新建时的状态为NEW
当FutureTask遇上DiscardPolicy,有坑
当FutureTask遇上DiscardPolicy,有坑
当FutureTask遇上DiscardPolicy,有坑

看到这里,一切正常,看不出什么问题,那么我们接着往下走。我们根据BUG触发条件可以知道,当线程池满触发拒绝策略时会出现问题,那么我们就去看看对应的源码。

以下为 ThreadPoolExecutor.execute(Runnable command) 的源码,红框圈起来的就是执行拒绝策略。
当FutureTask遇上DiscardPolicy,有坑

点进去可以看到,ThreadPoolExecutor 使用其 handler 执行拒绝策略。
当FutureTask遇上DiscardPolicy,有坑

当我们将handler设置成DiscardPolicy的时候,其rejectedExecution方法什么都没有执行,就是个空方法。
当FutureTask遇上DiscardPolicy,有坑

至此,从提交任务到拒绝任务再到获取任务结果,整个流程我们都走了一遍,我们发现了什么?是不是从开始到结束,FutureTask的状态都没有改变呢?一直都是新建时赋值的NEW状态,在这个状态下去获取执行结果是获取不到的,因为它还没有执行,所以就会一直等待它去执行,但是它其实已经被线程池拒绝掉了永远也不会执行了,so,卡死了。

这里又有个疑问了,那为什么默认拒绝策略AbortPolicy没有问题呢?答案是AbortPolicy拒绝时抛出了异常。这个异常会直接抛到主线程,所以需要在主线程捕获并处理这个异常,因为已经知道异常了,所以不会再去获取异步执行结果也就不会卡死了。
当FutureTask遇上DiscardPolicy,有坑
当FutureTask遇上DiscardPolicy,有坑

问题修复

既然知道导致问题的原因是线程池拒绝策略 DiscardPolicyDiscardOldestPolicy 导致的,那么就不用这两个策略嘛,如果一定要用的话,可以自己改造下,抛个异常出来可好?让主线程知道这个任务被拒绝了好进行下一步的操作,放弃也好,补偿也罢。

另外在获取异步结果的时候,可以加个超时时间保底,不至于让主线程阻塞。
当FutureTask遇上DiscardPolicy,有坑
修改后的结果,任务0和任务1成功执行,任务2、任务3、任务4被拒绝。
当FutureTask遇上DiscardPolicy,有坑

扩展

FutureTask 的状态流转是什么样的呢?
答案依然在源码里,在FutureTaskrun()方法中。
当FutureTask遇上DiscardPolicy,有坑

执行成功,最终状态为 NORMAL
当FutureTask遇上DiscardPolicy,有坑

出现异常时,最终状态为 EXCEPTIONAL
当FutureTask遇上DiscardPolicy,有坑

另外还可以通过cancel方法将状态改为 CANCELLED 或者 INTERRUPTEDINTERRUPTINGINTERRUPTED 的过渡状态。

只有状态为 NORMAL 时可以获取到执行结果,CANCELLEDINTERRUPTED 状态抛出CancellationException 异常,状态为 EXCEPTIONAL 时抛出 ExecutionException 异常。
当FutureTask遇上DiscardPolicy,有坑

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/155281.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

LVS+Keepalived+Nginx具体配置步骤

视频链接&#xff1a;4-6 搭建LVS-DR模式- 为两台RS配置虚拟IP_哔哩哔哩_bilibili 视频笔记链接&#xff1a;笔记 一、服务器与Ip约定 LVS DIP: 192.168.1.151 VIP: 192.168.1.150 Nginx1 RIP: 192.168.1.171 VIP: 192.168.1.150 Nginx2 RIP: 192.168.1.172 VIP: 192.168…

力扣 2283. 判断一个数的数字计数是否等于数位的值

题目 给你一个下标从 0 开始长度为 n 的字符串 num &#xff0c;它只包含数字。 如果对于 每个 0 < i < n 的下标 i &#xff0c;都满足数位 i 在 num 中出现了 num[i]次&#xff0c;那么请你返回 true &#xff0c;否则返回 false 。 示例 输入&#xff1a;num “1…

vue文件上传

vue文件上传 前言 今天写一篇关于文件上传的文章&#xff0c;其实对于真正工作的人来说&#xff0c;这不是很难的事&#xff0c;但对于新手来说无疑是个新技术 实现功能如下图 根据上传文件类型显示对应文件类型&#xff0c;图片直接显示&#xff0c;当然图片特可以做对应类型…

【NI Multisim 14.0原理图设计基础——参数属性设置】

目录 序言 &#x1f46c;一、参数属性设置 序言 NI Multisim最突出的特点之一就是用户界面友好。它可以使电路设计者方便、快捷地使用虚拟元器件和仪器、仪表进行电路设计和仿真。 首先启动NI Multisim 14.0&#xff0c;打开如图所示的启动界面&#xff0c;完成初始化后&…

初始C语言-分支与循环语句

目录 一、语句 1. 什么是语句&#xff1f; C语句可分为以下五类&#xff1a; 1. 表达式语句 2. 函数调用语句 3. 控制语句 4. 复合语句 5. 空语句 2. 控制语句&#xff1a; 1. 条件判断语句也叫分支语句&#xff1a;if语句、switch语句&#xff1b; 2. 循环执行语句…

extern “C“的作用以及c/c++代码互调

一、目的相信从事嵌入开发的小伙伴肯定遇到过使用第三库的情景&#xff0c;有时候可能是C中调用C库&#xff0c;有时候可能又是C中调用C库&#xff1b;如果你遇到过&#xff0c;那你肯定知道extern "C"的作用.本篇的目的就是给大家介绍C/C互相调用的原理和实践。二、…

RK3588平台开发系列讲解(内核调试篇)CPU Hotplug 调试

平台内核版本安卓版本RK3588Linux 5.10Android 12文章目录 一、CPU Hotplug 介绍二、CPU Hotplug 内核开关三、sysfs调试沉淀、分享、成长,让自己和他人都能有所收获!😄 📢多核系统底层驱动提供了cpu的Plugin/Unplug接口,可以实现动态调整cpu使用运行。 一、CPU Hotplu…

在 Node JS 中实现微服务架构

&#x1f4cd;简介&#x1f642; 正如我们在之前的博客“单体与微服务&#xff1a;一种实用方法”中讨论的那样。但是今天我们要在 NodeJS 中实现微服务架构。&#x1f449; 您可以使用任何技术&#xff0c;如 Spring、Python 等。但我们将使用 NodeJS 进行演示。&#x1f4cd;…

PostgreSQL下载、安装和配置使用

1&#xff09;下载 PostgreSQL官网下载&#xff1a;https://www.enterprisedb.com/downloads/postgres-postgresql-downloads 2&#xff09;安装 双击运行安装包 选择安装目录&#xff0c;不推荐安装在C盘。 选择需要安装的服务&#xff0c;不明白是啥的全选就行&#xf…

内含JAVA简单概括和JAVA所需安装的软件和详细教程,想学习JAVA无从下手,这篇文章带你迈出第一步

本文大致概括了JAVA编程语言的简史和特点,主要介绍了JAVA开发环境安装,涉及JDK,Sublime Text IntelliJ IDEA三个软件的简单介绍,安装和使用,最后编写了第一个JAVA代码,保姆级教学,跟着文章一步步来,迈出你学习JAVA的第一步吧! 初识JAVA一.JAVA语言简介二.JAVA发展简史三.JAVA语…

通过模拟器实现APP抓包

本教程将跳过工具安装部分&#xff0c;请正确食用&#x1f609;我的环境&#xff1a;操作系统&#xff1a;win11模拟器版本&#xff1a;雷电安卓模拟器稳定版-安卓7.1(32位) V5.0.46抓包工具&#xff1a;Wireshark-Version 4.0.1抓包工具&#xff1a;charles-Version 4.6.3工具…

SpringMVC(十一):SpringMVC文件上传中要解决的问题

文章目录 SpringMVC文件上传中要解决的问题 一、中文文件名编码问题 二、文件位置存储问题

【YOLO系列】YOLOv8算法(尖端SOTA模型)

前言回顾 在这里粗略回顾一下YOLOv5&#xff0c;这里直接提供YOLOv5的整理的结构图吧 Backbone&#xff1a;CSPDarkNet结构&#xff0c;主要结构思想的体现在C3模块&#xff0c;这里也是梯度分流的主要思想所在的地方&#xff1b;PAN-FPN&#xff1a;双流的FPN&#xff0c;必…

Linux——常用命令1.

✅<1>主页&#xff1a;我的代码爱吃辣 &#x1f4c3;<2>知识讲解&#xff1a;linux &#x1f525;<3>创作者&#xff1a;我的代码爱吃辣 ☂️<4>开发环境&#xff1a;Visual Studio 2022 &#x1f4ac;<5>前言&#xff1a;Linux常用命令 目录 …

Centos7安装JDK1.8(OracleJDK8)

Linux安装JDK1.8 | OracleJDK8Linux安装JDK1.8(OracleJDK8)卸载openJDK官网下载OracleJDK8解压JDK到指定目录配置JDK变量查看jdk是否可用Linux安装JDK1.8(OracleJDK8) Hadoop官方的JDK要求https://cwiki.apache.org/confluence/display/HADOOP/HadoopJavaVersions JDK变量配置…

Java——》下载Hotspot 虚拟机源码

推荐链接&#xff1a; 总结——》【Java】 总结——》【Mysql】 总结——》【Redis】 总结——》【Kafka】 总结——》【Spring】 总结——》【SpringBoot】 总结——》【MyBatis、MyBatis-Plus】 Java——》下载Hotspot 虚拟机源码一、官网下载压…

信息系统基础

目录 一、信息化概述 1.信息的定义 2.信息的特点 3.信息化的概念 4.信息对经济社会的意义 5.信息化对组织的意义 二、信息系统工程的总体规划 1.信息系统工程总体规划方法论 2.信息系统的生命周期和各阶段的目标 三.政府信息化与电子政务 1.电子政务的组成 2.电子政务…

mysql的binlog学习记录

文章目录什么是binlogbinlog格式StatementRowMixedbinlog使用什么是binlog MySQL Binary Log也就是常说的bin-log, ,是mysql执行改动产生的二进制日志文件。简单的来说&#xff0c;binlog日志用于记录所有更新了数据或者以及潜在更新了数据&#xff08;例如&#xff0c;没有匹…

在线学习(online learning)——Chapter 1 What is online learning

在线学习(online learning)——Chapter 1 What is online learning 引用 [1]Hoi S C H, Sahoo D, Lu J, et al. Online learning: A comprehensive survey[J]. Neurocomputing, 2021, 459: 249-289.点击阅读 [2]Orabona F. A modern introduction to online learning[J]. ar…

【1分钟了解网络交换机的6种命令配置模式】

我们在配置交换机的时候首先要了解的就是交换机命令模式&#xff0c;小编用Cisco思科交换机为例带大家了解交换机的6种配置模式。 Cisco IOS提供了用户EXEC模式和特权EXEC模式两种基本的命令执行级别&#xff0c;同时还提供了全局配置、接口配置、Line配置和vlan数据库配置等多…