Spring统一异常处理捕获不到CompletableFuture异步编排中的异常的问题

news2024/11/14 15:29:10

Spring统一异常处理捕获不到CompletableFuture异步编排中的异常的问题

  • Spring统一异常处理捕获不到CompletableFuture异步编排中的异常的问题
    • Spring统一异常处理
    • 简单例子
    • 多线程无法捕获场合
      • 正常场合(不使用异步编排)代码
        • ServiceIm
        • ExcpHandler
        • Controller
        • 结果
      • 异常场合(使用异步编排)代码
        • Controller
        • 结果
      • 使用join方法阻塞
        • 修改
        • 结果
      • 使用join的问题(不使用exceptionally捕获)
    • 结论

Spring统一异常处理捕获不到CompletableFuture异步编排中的异常的问题

Spring统一异常处理

我们常常使用@RestControllerAdvice@ControllerAdvice配合@ExceptionHandler来做全局的统一异常处理,这种方式被认为是十分高效的

简单例子

/**
 * @author Syf200208161018
 * @date 2022/12/3 15:21
 * @ClassName:ExcpHandler
 * @Effect:ExcpHandler is used for
 */
@RestControllerAdvice(annotations = {RestController.class})
public class ExcpHandler {

    @ExceptionHandler(value = {Exception.class})
    public String deal(Exception e){
        System.out.println(e.getMessage());
        return e.getMessage();
    }
}

在这个例子中,只要Controller中(带有@RestController注解)出现任意异常就会被捕捉到!
但是我们需要明确的是,我上面这句话只在主线程的情况下成立!

多线程无法捕获场合

正常场合(不使用异步编排)代码

在这里插入图片描述

ServiceIm

在这个类中我们看到有一个int i = 1/0;表明一定会引发除0异常!

/**
 * @author Syf200208161018
 * @date 2022/12/3 15:23
 * @ClassName:Service
 * @Effect:Service is used for
 */
@Service
public class ServiceIm {
    public void test(){
        int i = 1/0;
    }
}

ExcpHandler

/**
 * @author Syf200208161018
 * @date 2022/12/3 15:21
 * @ClassName:ExcpHandler
 * @Effect:ExcpHandler is used for
 */
@RestControllerAdvice(annotations = {RestController.class})
public class ExcpHandler {

    @ExceptionHandler(value = {Exception.class})
    public String deal(Exception e){
        System.out.println(e.getMessage());
        return e.getMessage();
    }
}

Controller

/**
 * @author Syf200208161018
 * @date 2022/12/3 15:22
 * @ClassName:Controller
 * @Effect:Controller is used for
 */
@RestController
public class Controller {
    @Autowired
    private ServiceIm serviceIm;

    @GetMapping("/test")
    public String test(){
         serviceIm.test();
        return "ok";
    }
}

结果

可以看到结果很正常,确实把错误返回了
在这里插入图片描述

异常场合(使用异步编排)代码

我们只要把Controller中的代码改成如下:

Controller

/**
 * @author Syf200208161018
 * @date 2022/12/3 15:22
 * @ClassName:Controller
 * @Effect:Controller is used for
 */
@RestController
public class Controller {
    @Autowired
    private ServiceIm serviceIm;

    @GetMapping("/test")
    public String test(){
        CompletableFuture.runAsync(() -> {
            serviceIm.test();
        }).exceptionally((e) -> {
            throw new RuntimeException("cuowu");
        });
        
        return "ok";
    }
}

结果

可以看到结果是ok,而且查看控制台并没有进行异常捕获,有些人可能说是因为已经使用了异步编排的exceptionally做了异常处理,可是我们要知道我在里面是直接抛出RuntimeException的!,那去除异步编排的异常处理可以吗?答案是也无法捕获到,更本原因就是并不发生在主线程上,或者说,主线程已经执行到了return但是实际我们的业务并没有结束
在这里插入图片描述

使用join方法阻塞

我们在学习异步编排的时候有一个join方法不知道大家记不记得
用于对任务线进行阻塞,join和get都可以,看场景,单纯阻塞无返回值建议join

在这里插入图片描述

修改

@RestController
public class Controller {
    @Autowired
    private ServiceIm serviceIm;

    @GetMapping("/test")
    public String test(){
        final CompletableFuture<Void> thread1 = CompletableFuture.runAsync(() -> {
            serviceIm.test();
        }).exceptionally((e) -> {
            throw new RuntimeException("cuowu");
        });
        thread1.join();

        return "ok";
    }
}

结果

结果我们发现成功了!当然这是意料之中的!因为这时候主线程就阻塞等待异步编排结束,异常也就被捕获到了
在这里插入图片描述

使用join的问题(不使用exceptionally捕获)

虽然join可以解决我们的问题,但是阻塞带来的问题就是性能变低,效率变低,我们使用异步编排的目的就是为了提升效率,所以若不用exceptionally进行捕获就会导致效率降低的问题

结论

如此我们得到一个结论,使用异步编排的正确姿势:

  1. 构建线程池
  2. 任务结束使用exceptionally进行捕获
  3. 使用join阻塞防止异常捕获失效
  4. 使用统一异常处理捕获

但是注意这只是对于时效性强必须有返回值的方法才能这样做!!!
也就是对于不要求及时返回的不要这样干,这里我们建议使用RabbitMQ处理!!!或者简单的使用@Async

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

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

相关文章

2023最新SSM计算机毕业设计选题大全(附源码+LW)之java高校车辆租赁管理系统23qhn

要开始我们毕业设计的第一步的关键就是选好我们的课题&#xff0c;有的同学开始选题的时候想着按照传统的课题延续下去&#xff0c;在设计题目时&#xff0c;不要过于笼统广泛&#xff0c;选择题目其实并不难&#xff0c;要多从自身的角度出发&#xff0c;要结合你们当前所处的…

【java】网络编程

文章目录网络编程概述基本概念IP地址概念InetAddress端口与协议概念UDP通信编程UDP发送数据UDP接受数据UDP通信程序练习TCP通信编程TCP发送数据TCP接收数据TCP通信程序练习网络编程概述 基本概念 IP地址概念 终端检查&#xff1a; InetAddress package heima.网络编程;impor…

nginx配置文件 location语法

1&#xff1a;nginx官方文档给出location语法如下&#xff1a; location [|~|~*|^~] uri { ....... }2&#xff1a;路径匹配 开头表示精确匹配。如 A 中只匹配根目录结尾的请求&#xff0c;后面不能带任何字符串&#xff1b;^~ 开头表示uri以某个常规字符串开头&#xff0c;不是…

字符串中第二大的数字(遍历)

力扣链接&#xff1a;力扣 给你一个混合字符串 s &#xff0c;请你返回 s 中 第二大 的数字&#xff0c;如果不存在第二大的数字&#xff0c;请你返回 -1 。 混合字符串 由小写英文字母和数字组成。 示例 1&#xff1a; 输入&#xff1a;s "dfa12321afd" 输出&…

Keras生成式学习(五)

生成式深度学习 生成式学习即创造学习&#xff0c;深度学习开始创造 一、使用LSTM 生成文本 给定前面的标记&#xff08;token&#xff0c;通常是单词或字符&#xff09;&#xff0c;能够对下一个标记的概率进行建模的任何网络都叫作语言模型&#xff08;language model&…

Android Camera性能分析 - 第25讲 CameraServer LatencyHistogram简介

本讲是Android Camera性能分析专题的第25讲 ​&#xff0c;我们介绍CameraServer LatencyHistogram简介&#xff0c;包括如下内容&#xff1a; LatencyHistogram是什么CameraServer默认有哪些Latency的Histogram如何获取CameraLatencyHistogram数据CameraLatencyHistogram类详…

Java项目:SSM CRM人事管理系统

作者主页&#xff1a;源码空间站2022 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文末获取源码 项目介绍 CRM人事管理系统&#xff0c;主要功能有&#xff1a; 用户管理&#xff1a;用户查询、添加用户、编辑、删除&#xff1b; 职位管理&#xff1a…

Java基于springboot+vue的防护用品销售购物商城系统 前后端分离

开发背景 随着近些年疫情的爆发人们对个人医疗相关防护也越来越重视了&#xff0c;尤其是在疫情开始之初&#xff0c;人们对疫情感受到非常的恐慌&#xff0c;虽然在国家和领导人的领导下疫情一次次的得到了控制&#xff0c;但是我们还是要做好个人防护&#xff0c;为了让更多…

5、Linux文件系统

目录 1、万事万物皆文件 2、Linux二级文件目录 3、Linux的文件操作 4、读取文件信息 在Linux中万物皆文件 1、万事万物皆文件 1)在Linux中&#xff0c;所有的东西都是以文件的方式进行操作 2)在Linux中&#xff0c;文件的访问和Window的不一样。window依靠的是通过盘符进…

Tomcat经验2

背景 资产系统与财务系统对接&#xff0c;开发经常让我在数据库中执行数据库更新语句&#xff0c;并重启Tomcat&#xff0c;将log文件发给他&#xff0c;他能够根据log文件判断&#xff0c;数究竟有没有传对。 在这个过程中会出现一个问题&#xff1a; 部署在同一台Web服务器…

ORT(ONNX runtime)GPU 问题总结

现象 bt的堆栈信息 standard io上的错误输出 从报错信息上看是非法的内存访问&#xff0c;但是报错的位置不一定是真实的位置&#xff0c;因为GPU都是异步发起的&#xff0c;错误可能会被在后面的op捕捉。例如cudaEventDestory&#xff1a; debug方式 思维方式 复现&#…

Python中matplotlib为多个列表数据绘制小提琴图

本文介绍基于Python中matplotlib模块与seaborn模块&#xff0c;利用多个列表中的数据&#xff0c;绘制小提琴图&#xff08;Violin Plot&#xff09;的方法。 小提琴图作为一种将箱型图与核密度图分别所能表达的信息相结合的数据可视化图&#xff0c;在数据分析中得以广泛应用&…

【电动车】电动汽车两阶段优化调度策略(Matlab代码实现)

目录 1 概述 2 数学模型 3 运行结果 4 结论 4 Matlab代码实现 1 概述 在当前阶段,电动汽车通常被视为即插即充的常规负荷,这浪费了其可观的储荷能力"。研究表明,非快充需求的电动汽车停靠时间远大于其充电时间[2,因此可通过充电站管理电动汽车的充放 电功率,使电动…

数据分析必备的5个工具,你用过几个?

数据有了&#xff0c;但是怎么让它们“听话”且输出为好看的图表呢&#xff1f;相信这是每一个做数据的人都思考过的问题吧。我们在工作、学习中经常进行会借助一些表格或者图表来进行信息的展示&#xff0c;一般大家最常用的是Excel表格或者PPT&#xff0c;但是随着近些年物联…

震惊,一个csdn小编用Python语言写了一个足球游戏,成功模拟世界杯决赛现场

前言 halo&#xff0c;包子们下午好 最近世界杯不是很火呀 很多小伙伴应该都知道球赛反正买&#xff0c;别墅靠大海&#xff01; 今天就给大家实现一个类似世界杯的足球小游戏&#xff0c;咱就说真的堪比国足了&#xff01; 哈哈哈~ 好啦 直接开整&#xff01;&#xff01;&am…

39-65-javajvm-运行时数据区-pc-栈

39-javajvm-运行时数据区&#xff1a; 运行时数据区 内存是非常重要的系统资源&#xff0c;是硬盘和CPU的中间仓库及桥梁&#xff0c;承载着操作系统和应用程序的实时运行JVM内存布局规定了Java在运行过程中内存申请、分配、管理的策略&#xff0c;保证了JVM的高效稳定运行。不…

支付宝支付项目

文章目录&#x1f68f; 支付宝支付项目&#x1f680; 支付宝介绍&#x1f6ac; 1、支付宝平台&#x1f6ac; 2、支付宝开放平台&#x1f6ac; 3、支付能力&#x1f6ad; 条码支付应用场景&#x1f6ad; 扫码支付应用场景&#x1f6ad; App 支付&#x1f6ad; 手机网站支付&…

【CSS3】media,伪类与伪元素,outline,font-face,resize,svg,多列布局

❤️ Author&#xff1a; 老九 ☕️ 个人博客&#xff1a;老九的CSDN博客 &#x1f64f; 个人名言&#xff1a;不可控之事 乐观面对 &#x1f60d; 系列专栏&#xff1a; 文章目录media例子resolution属性orientation属性aspect-ratio属性伪类与伪元素:target:disable/:enable:…

[附源码]计算机毕业设计校园生活服务平台Springboot程序

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

【SQL教程|01】SQL简介——什么是SQL

什么是SQL SQL是一门语言 SQL是Structured Query Language的简写&#xff0c;中文译为“结构化查询语言”&#xff1b;SQL是一种用来查询和处理关系性数据库的语言&#xff0c;使用SQL我们可以&#xff1a; 增&#xff08;INSERT&#xff09;&#xff1a;可以向数据库中插入记…