异常-捕获业务异常踩坑记录

news2025/1/16 4:54:19

事情是这样,用了google的一个本地缓存框架,就是在查询数据的时候如果有就取缓存,没有就发http请求调接口,但是http请求也会有查询失败的时候,查询失败就会手动抛一个业务异常,然后我会在外层各种捕获异常,通过判断异常类型来做特殊处理。结果发现,手动抛了一个异常,但是catch里面没有找到我的异常,原因是什么下面和大家分享下,顺便记录下自己的踩坑日志。

通过LoadingCache来获取内容,里面手动抛了DomainServiceException异常,但是在捕获的时候却进到了Exception这是为什么?

先说下结论:LoadingCache会把我们定义的业务异常转成ExecutionException,最后又转成了UncheckedExecutionException了,但是无论怎么转,我们的业务异常都在cause里面。

二、下面讲下LoadingCache是如何实现的

当执行LoadingCache的get方式时会进到loadFuture方法,里面判断本地缓存是否存在不存在就发http请求查询数据

1、loadFuture方法

public ListenableFuture<V> loadFuture(K key, CacheLoader<? super K, V> loader) {
    try {
        
        V previousValue = oldValue.get();
        if (previousValue == null) {
            V newValue = loader.load(key);  //1、如果缓存为空,执行具体的目标方法,来获取数据。
            return set(newValue) ? futureValue : Futures.immediateFuture(newValue);
        }
        ListenableFuture<V> newValue = loader.reload(key, previousValue);
        if (newValue == null) {
            return Futures.immediateFuture(null);
        }
        
        return transform(
                newValue,
                new com.google.common.base.Function<V, V>() {
                    @Override
                    public V apply(V newValue) {
                        LocalCache.LoadingValueReference.this.set(newValue);
                        return newValue;
                    }
                },
                directExecutor());
    } catch (Throwable t) {
        ListenableFuture<V> result = setException(t) ? futureValue : fullyFailedFuture(t);
        if (t instanceof InterruptedException) {
            Thread.currentThread().interrupt();
        }
        return result;
    }
}

因为http请求里面做了业务处理,如果没有查询到,就会报错,也就是loader.load这段代码会报错,但是上面的代码有try-catch来捕获异常,然后得到Failure对象,Failure对象里面保存了具体的业务异常,最终Failure对象被存到了SettableFuture类型对象里面。看下SettableFuture里面什么样,如下图。然后返回SettableFuture对象。

1、loadSync方法

V loadSync(K key, int hash, LoadingValueReference<K, V> loadingValueReference, CacheLoader<? super K, V> loader) throws ExecutionException {
    //查询数据,也包括从缓存获取内容,如果报错,报错对象也会在loadingFuture对象里面。
    ListenableFuture<V> loadingFuture = loadingValueReference.loadFuture(key, loader); 

    //解析loadingFuture,如果成功会把数据存到本地缓存里面,如果失败,就会把错误抛出来。
    return getAndRecordStats(key, hash, loadingValueReference, loadingFuture);
}

因为loadFuture报错了,但是还是返回了ListenableFuture对象,getAndRecordStats会解析ListenableFuture对象。

//根据对象类型来判断。是报错还是缓存数据
private V getDoneValue(Object obj) throws ExecutionException {
    if (obj instanceof AbstractFuture.Cancellation) {
        throw cancellationExceptionWithCause("Task was cancelled.", ((AbstractFuture.Cancellation) obj).cause);
    } else if (obj instanceof AbstractFuture.Failure) {
        throw new ExecutionException(((AbstractFuture.Failure) obj).exception);  //注意这里会把错误转成Google自定义的异常,我们的业务异常在cause里面。
    } else if (obj == NULL) {
        return null;
    } else {
        V asV = (V) obj;
        return asV;
    }
}

然后转成ExecutionException错误,在抛出来,但是new ExecutionException的时候已经把我么自定义的业务场景,当做参数传给了ExecutionException。

最后异常被下面代码捕获

V get(K key, int hash, CacheLoader<? super K, V> loader) throws ExecutionException {
    try {
        return lockedGetOrLoad(key, hash, loader);
    } catch (ExecutionException ee) {
        Throwable cause = ee.getCause();   //拿到业务异常,判断是否Error还是RuntimeException
        if (cause instanceof Error) {
            throw new ExecutionError((Error) cause);
        } else if (cause instanceof RuntimeException) {
            throw new UncheckedExecutionException(cause);
        }
        throw ee;
    } finally {
        postReadCleanup();
    }
}

因为我们自定义的异常都是继承了RuntimeException,所以会走到UncheckedExecutionException,然后转成UncheckedExecutionException异常。

结论:我们业务自定义的异常DomainServiceException先被Google转成了ExecutionException,最后又转成了UncheckedExecutionException了,但是无论怎么转,我们的业务异常都在cause里面。

所以我们在判断业务异常,不能直接使用下图这种方式获取

而是要使用下图这种方式

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

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

相关文章

三消游戏查找算法的原理和实现

本文首发于公众号&#xff1a; 小蚂蚁教你做游戏。欢迎关注领取更多学习做游戏的原创教程资料&#xff0c;每天学点儿游戏开发知识。嗨&#xff01;大家好&#xff0c;我是小蚂蚁。今天这篇文章分享一下三消查找算法的原理和实现&#xff0c;其实三消的机制最早源于《宝石方块》…

无联不成春,2023创宇网安春联展!

春 联 又叫“春贴”“门对”“对联” 它以对仗工整、简洁精巧的文字 描绘美好形象&#xff0c;寄托美好愿望 是中国独特的文学形式 网络安全的列车已驶入2023&#xff0c;癸卯兔年新春将至&#xff01; 热爱中华优秀传统文化的创宇人&#xff0c;也纷纷以春联为载体&…

大网进阶安全刷题讲解(带答案)(1)

作者简介&#xff1a;一名云计算网络运维人员、每天分享网络与运维的技术与干货。 座右铭&#xff1a;低头赶路&#xff0c;敬事如仪 个人主页&#xff1a;网络豆的主页​​​​​​ 目录 前言 一.大网进阶安全刷题 前言 本章将会讲解大网进阶安全刷题讲解。 一.大网进阶…

测试开发 | AppCrawler 自动遍历测试实践(二):定制化配置

本文为霍格沃兹测试学院学院学员课程 AppCrawler 学习笔记&#xff0c;文末加群一起学习交流。 定制化配置 自动遍历测试技术以及工具该如何选择和快速入门&#xff1f;经过对比和需求&#xff0c;最终选择测试架构师思寒大佬的 AppCrawler 作为自动遍历测试的工具。以下就分享…

LeetCode 112. 路径总和

&#x1f308;&#x1f308;&#x1f604;&#x1f604; 欢迎来到茶色岛独家岛屿&#xff0c;本期将为大家揭晓LeetCode 112. 路径总和&#xff0c;做好准备了么&#xff0c;那么开始吧。 &#x1f332;&#x1f332;&#x1f434;&#x1f434; 一、题目名称 LeetCode 112…

Cadence PCB仿真使用Allegro PCB SI按照指定的规则自动创建差分对方法图文教程

⏪《上一篇》   🏡《总目录》   ⏩《下一篇》 目录 1,概述2,配置方法3,总结1,概述 本文简单介绍使用Allegro PCB SI按照指定的规则自动创建差分的方法。 2,配置方法 第1步:打开待仿真的PCB文件,并确认软件为Allegro PCB SI 如果,打开软件不是Allegro PCB SI则可…

在使用示波器时,为什么有些波形感觉一直在晃?

我们可以从三个角度进行分析&#xff1a; 采样分析&#xff1a; 1、示波器采样信号的过程如上图&#xff1a;采样——处理——采样——处理。 2、处理时间也称死区时间&#xff0c;死区时间内示波器不监测输入信号。 3、提高波形刷新率&#xff0c;实质上是减少了死区时间&…

MyBatis-Plus数据安全保护(配置安全)

SpringBootMyBatis-Plus配置安全 1.该功能为了保护数据库配置及数据安全&#xff0c;在一定的程度上控制开发人员流动导致敏感信息泄露2.加密配置 mpw: 开头紧接加密内容&#xff08; 非数据库配置专用 YML 中其它配置也是可以使用的 &#xff09;3.随机密钥请负责人妥善保管&a…

设计模式_创建型模式 -《原型模式》

设计模式_创建型模式 -《原型模式》 笔记整理自 黑马程序员Java设计模式详解&#xff0c; 23种Java设计模式&#xff08;图解框架源码分析实战&#xff09; 概述 原型模式 (Prototype Pattern) &#xff1a;用一个已经创建的实例作为原型&#xff0c;通过复制该原型对象来创建…

Kafka的相关知识

一. Kafka基本介绍 Kafka是一个分布式、支持分区的&#xff08;partition&#xff09;、多副本的&#xff08;replica&#xff09;&#xff0c;基于zookeeper协调的分布式消息系统。具有&#xff1a;高吞吐量、低延迟、可扩展性、持久性、可靠性、容错性、高并发等特性。常见的…

RabbitMQ(五)常见面试题

目录1. 什么是 RabbitMQ&#xff1f;2.为什么要使用 RabbitMQ&#xff1f;3.使用 RabbitMQ 的场景4.如何确保消息正确地发送至 RabbitMQ&#xff1f;如何确保消息接收方消费了消息&#xff1f;5.如何避免消息重复投递或重复消费&#xff1f;6.消息基于什么传输&#xff1f;7.消…

基于java ssm springboot网上蛋糕商城项目设计和实现

基于java ssm springboot网上蛋糕商城项目设计和实现 博主介绍&#xff1a;5年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 超级帅帅吴 Java毕设项目精品实战案例《500套》 欢迎点赞 收藏 ⭐留言 文末获取源…

CAD中还能装ArcGIS?还能加载影像?定义坐标系?

时常我们需要CAD中插入影像 需要CAD数据转换GIS数据 需要CAD加载GIS数据 需要将CAD与GIS数据套库 却被坐标系搞蒙了 今天&#xff0c;我们就来解决以上的问题&#xff0c;介绍一款免费的ArcGIS插件——ArcGIS for AutoCAD。 这个模块插件可以直接访问在线地图。直接在CAD中…

Java实验——定义一个表示学生信息的类Student,要求如下:

一、题目要求 定义一个表示学生信息的类Student&#xff0c;要求如下&#xff1a; &#xff08;1&#xff09;类Student的成员变量&#xff1a; sNO 表示学号&#xff1b;sName表示姓名&#xff1b;sSex表示性别&#xff1b;sAge表示年龄&#xff1b;sJava&#xff1a;表示…

测试开发 | 如何模拟真实使用场景?mock 技术来帮你

mock 是一种通过代理修改请求与响应&#xff0c;从而辅助构造更多应用场景的工具。比如在工作中&#xff0c;可能需要 mock 第三方的回调给到测试人员测试的环境&#xff0c;从而更顺利的开展测试工作&#xff0c;也使得测试环境更接近真实的使用场景。Charles 修改请求与响应M…

Android自定义时间选择器

效果图 一、添加NumberPicker开源库 需要添加以下控件 仓库地址&#xff1a; https://github.com/ShawnLin013/NumberPicker implementation io.github.ShawnLin013:number-picker:2.4.13 二、 添加弹出框主题样式 在drawable文件夹下新建一个bg_bottom_dialog.xml&…

jenkins结合gitlable企业集成部署实战

简介 Jenkins是一个开源软件项目&#xff0c;起源于Hudson&#xff08;Hudson是商用的&#xff09;&#xff0c;是基于Java开发的一种持续集成工具&#xff0c;用于监控持续重复的工作&#xff0c;旨在提供一个开放易用的软件平台&#xff0c;使软件项目可以进行持续集成 1.0 …

新的跨平台渲染引擎:Ab3d.DXEngine 5.2 Crack

用于 WPF 和 WinForms 的 DirectX 11 3D 渲染引擎 Ab3d.DXEngine 是一种超快的 3D 渲染引擎&#xff0c;可用于 .Net 桌面应用程序。采集 by Ω578867473 Ab3d.DXEngine采用超快的多线程渲染技术&#xff0c;可以充分利用显卡&#xff0c;提供与使用C时几乎相同的极致性能。 该…

javaEE 初阶 — 线程安全的集合类

文章目录1. 多线程环境使用 ArrayList多线程使用队列3. 多线程环境使用哈希表3.1 Hashtable3.2 ConcurrentHashMap4. 相关面试题1. 多线程环境使用 ArrayList java 标准库里的大部分集合类都是 “线程不安全” 的。 多个线程使用同一个集合类对象&#xff0c;很有可能会出问题。…

WebDAV之葫芦儿·派盘+Air Explorer

Air Explorer 支持WebDAV方式连接葫芦儿派盘。 电脑本地硬盘不够,网盘云存储已经成为存储的一种趋势。网盘限额,数据存在多个网盘内,操作麻烦还费事。 Airexplorer可以完美解决这个困扰。Air Explorer是一款非常不错的云存储账户管理软件,常见的在线云存储服务器都支持;还…