当前位置: 首页 > news >正文

SpringBoot异常处理

目录

一、 错误处理

1. 默认规则

2. 定制错误处理逻辑

二、自定义异常处理

1. 实现 ErrorController

2. @RestControllerAdvice/@ControllerAdvice + @ExceptionHandler 实现自定义异常

3. 新建 UserController.class 测试 3 种不同异常的处理

4. 最终效果如下

补充

1. 参数校验所需依赖以及使用方式

2. 追踪抛出错误的方法


一、 错误处理

1. 默认规则

1)默认情况下,SpringBoot 提供了 /error 处理所有错误的映射

2)对于机器客户端,它将生成 JSON 响应,其中包含错误,HTTP 状态和异常消息的详细信息;对于浏览器客户端,它将响应一个 "whitelabel" 错误视图,以 HTML 格式呈现相同的数据

3)要对其自定义,添加 View 解析为 Error

4)要完全替换默认行为,可以实现 ErrorController 并注册该类型的 Bean 定义,或添加 ErrorAttributes 类型组件以使用现有机制并替换其内容;ErrorAttributes 中定义了返回哪些错误项

2. 定制错误处理逻辑

1)自定义错误页面 error/404.html,error/500.html 放到 /resources/public/error 文件夹下即可;也可以将这两个 html 命名为 4xx.html,5xx.html 则可处理对应 4xx、5xx 错误;当然还可以每个状态码放一个对应页面

 

2)通过 @ControllerAdvice/@RestControllerAdvice + @ExceptionHandler 处理异常

3)实现 HandlerExceptionResolver 处理异常

二、自定义异常处理

1. 实现 ErrorController

1. 新建 GlobalErrorController.class 实现 ErrorController 接口并实现如下逻辑

@RestController
public class GlobalErrorController implements ErrorController {// 错误请求路径private final String ERROR_PATH = "/error";@Resourceprivate ErrorAttributes errorAttributes;@Overridepublic String getErrorPath() {return ERROR_PATH;}/*** JSON格式错误信息*/@RequestMapping(value = ERROR_PATH,  produces = {MediaType.APPLICATION_JSON_VALUE})public Map<String, Object> error(WebRequest webRequest) {ErrorAttributeOptions options = ErrorAttributeOptions.of(ErrorAttributeOptions.Include.MESSAGE);Map<String, Object> body = this.errorAttributes.getErrorAttributes(webRequest, options);return body;}
}

注意

1)ERROR_PATH = "/error" 这个路径其实就是 SpringBoot 错误处理机制中自动配置的路径;在 ErrorProperties.class 中可找到这个配置

2)如果在 application.properties 覆盖了默认错误路径,则上面代码中 ERROR_PATH 应设置为配置文件中的错误路径

# 全局错误映射路径
server.error.path = /error

3)这句代码:ErrorAttributeOptions.of(ErrorAttributeOptions.Include.MESSAGE);的意思是返回给前端的错误信息中包含哪些信息,这个 of() 方法中可以指定返回信息范围(可以指定多个),共有 4 个选项值:

  • ErrorAttributeOptions.Include.EXCEPTION:Include the exception class name attribute
  • ErrorAttributeOptions.Include.STACK_TRACE:Include the stack trace attribute
  • ErrorAttributeOptions.Include.MESSAGE:Include the message attribute
  • ErrorAttributeOptions.Include.BINDING_ERRORS:Include the binding errors attribute

不同值返回信息如下

a. ErrorAttributeOptions.Include.EXCEPTION

{"timestamp": "2022-08-07T13:53:40.607+00:00","status": 500,"error": "Internal Server Error","exception": "java.lang.RuntimeException","message": "","path": "/error123"
}

b. ErrorAttributeOptions.Include.STACK_TRACE,显示信息最多

{"timestamp": "2022-08-07T13:54:14.101+00:00","status": 500,"error": "Internal Server Error","trace": "java.lang.RuntimeException: Error\r\n\tat com.study ... ..."message": "","path": "/error123"
}

c. ErrorAttributeOptions.Include.MESSAGE

{"timestamp": "2022-08-07T13:54:56.751+00:00","status": 500,"error": "Internal Server Error","message": "Error","path": "/error123"
}

d. ErrorAttributeOptions.Include.BINDING_ERRORS

{"timestamp": "2022-08-07T13:53:03.791+00:00","status": 500,"error": "Internal Server Error","message": "","path": "/error123"
}

2. @RestControllerAdvice/@ControllerAdvice + @ExceptionHandler 实现自定义异常

在一次请求调用过程中,如果程序出现异常,我们应对异常进行拦截,把异常中重要信息记录在日志便于排查错误,只提供简单的错误信息返回给前端;这个时候通过 @RestControllerAdvice/@ControllerAdvice + @ExceptionHandler 可轻松实现

1. 新建 GlobalException.class 继承 RuntimeException 并重写构造方法

@Data
public class GlobalException extends RuntimeException {// 错误编码private Integer errorCode;// 错误信息private String errorMsg;// 具体信息private String errorInfo;public GlobalException(Integer errorCode, String errorMsg, String errorInfo) {super(errorMsg);this.errorCode = errorCode;this.errorMsg = errorMsg;this.errorInfo = errorInfo;}public GlobalException(Integer errorCode, String errorMsg, Throwable cause) {super(errorMsg, cause);this.errorCode = errorCode;this.errorMsg = errorMsg;this.errorInfo = cause.getMessage();}
}

2. 新建 GlobalExceptionHandler.class 实现对系统错误的统一处理

@RestControllerAdvice
public class GlobalExceptionHandler {/*** 处理全局系统异常(非自定义异常)*/@ExceptionHandler(Exception.class)public Map<String, Object> handleSystemException(Exception exception) {Map<String, Object> exceptionInfo = new HashMap<>();exceptionInfo.put("errorCode", 500);exceptionInfo.put("errorMsg", "系统故障");// 日志记录具体信息,不要返回给前端System.out.println(exception.getMessage());return exceptionInfo;}/*** 处理自定义异常*/@ExceptionHandler(GlobalException.class)public Map<String, Object> handleGlobalException(GlobalException exception) {Map<String, Object> exceptionInfo = new HashMap<>();exceptionInfo.put("errorCode", exception.getErrorCode());exceptionInfo.put("errorMsg", exception.getErrorMsg());// 日志记录具体信息,不要返回给前端System.out.println(exception.getErrorInfo());return exceptionInfo;}/*** 实体类属性校验异常*/ @ExceptionHandler(MethodArgumentNotValidException.class)public Map<String, Object> MethodArgumentNotValidExceptionHandler(MethodArgumentNotValidException exception) {ObjectError objectError = exception.getBindingResult().getAllErrors().get(0);Map<String, Object> exceptionInfo = new HashMap<>();exceptionInfo.put("errorCode", 550);exceptionInfo.put("errorMsg", objectError.getDefaultMessage());return exceptionInfo;}
}

3. 新建 UserController.class 测试 3 种不同异常的处理

@RestController
public class UserController {/*** 测试全局异常*/@GetMapping("/error1")public Integer error1() {int m = 10 / 0;return m;}/*** 测试自定义异常*/@GetMapping("/error2")public Integer error2() {try {int m = 10 / 0;return m;} catch (Exception e) {throw new GlobalException(501, "系统错误", e);}}/*** 测试实体类字段校验异常*/@PostMapping("/error3")public User error3(@RequestBody @Valid User user) {return user;}
}

4. 最终效果如下

1)全局异常测试(非自定义异常):http://localhost:8080/error1

{"errorCode": 500,"errorMsg": "系统故障"
}

2)自定义异常测试:http://localhost:8080/error2

{"errorCode": 501,"errorMsg": "系统错误"
}

3)实体类字段校验异常测试:http://localhost:8080/error3

{"errorCode": 550,"errorMsg": "年龄不能超过100岁"
}

补充

1. 参数校验所需依赖以及使用方式

1)实体类字段校验需要引入如下依赖

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-validation</artifactId>
</dependency>

2)实体类上添加注解

@Data
public class User {@NotBlank(message = "用户名不能为空")private String name;@NotNull(message = "年龄不能为空")@Min(value = 1, message = "年龄不能小于1岁")@Max(value = 100, message = "年龄不能超过100岁")private Integer age;
}

2. 追踪抛出错误的方法

上面的异常处理,虽然把返回给前端的异常提示信息和日志记录的异常信息进行了处理,但对于后端来说,日志仅仅打印了错误信息,并没有记录是哪个类哪个方法的抛出的异常,这不便于后端调试,所以可添加如下方法记录抛出错误的类和方法

1)添加根据异常查找出执行的类和方法

/*** 找出执行类和执行方法*/
private String getExecutedMethod(Exception e) {StackTraceElement[] elements = e.getStackTrace();if(elements.length > 0) {// 异常链中第一个也就是最外层的信息, 当然就是 controller 这一层StackTraceElement target = elements[0];return String.format("%s#%s", target.getClassName(), target.getMethodName());}return "";
}

2)在异常处理器中做如下修改即可

/*** 处理全局系统异常(非自定义异常)*/
@ExceptionHandler(Exception.class)
public Map<String, Object> handleSystemException(Exception exception) {Map<String, Object> exceptionInfo = new HashMap<>();exceptionInfo.put("errorCode", 500);exceptionInfo.put("errorMsg", "系统故障");// 日志记录具体信息,不要返回给前端String executedMethod = getExecutedMethod(exception);String exceptionMessage = String.format("执行方法: %s, 错误信息: %s", executedMethod, exception.getMessage());tem.out.println(exceptionMessage);return exceptionInfo;
}

相关文章:

SpringBoot异常处理

目录 一、 错误处理 1. 默认规则 2. 定制错误处理逻辑 二、自定义异常处理 1. 实现 ErrorController 2. RestControllerAdvice/ControllerAdvice ExceptionHandler 实现自定义异常 3. 新建 UserController.class 测试 3 种不同异常的处理 4. 最终效果如下 补充 1. 参…...

《C++ Primer Plus》(第6版)第8章编程练习

《C Primer Plus》&#xff08;第6版&#xff09;第8章编程练习《C Primer Plus》&#xff08;第6版&#xff09;第8章编程练习1. 打印字符串2. CandyBar3. 将string对象的内容转换为大写4. 设置并打印字符串5. max5()6. maxn()7. SumArray()《C Primer Plus》&#xff08;第6版…...

RAD Studio 11.3 Alexandria Crack

RAD Studio 11.3 Alexandria Crack 瞄准最新平台版本-此版本增加了对Android 13和Apple macOS Ventura的官方支持。它还支持Ubuntu 22 LTS和Microsoft Windows Server 2022。 使用生物特征认证-New为FireMonkey移动应用程序提供了新的移动生物特征认证组件。 部署嵌入式InterBa…...

Stm32 iic 协议使用

/* 第1个参数为I2C操作句柄 第2个参数为从机设备地址 第3个参数为从机寄存器地址 第4个参数为从机寄存器地址长度 第5个参数为发送的数据的起始地址 第6个参数为传输数据的大小 第7个参数为操作超时时间 */ HAL_I2C_Mem_Write(&hi2c2,salve_add,0,0,PA_BUFF,sizeof(PA_BUFF…...

Malware Dev 02 - Windows SDDL 后门利用之 SCManager

写在最前 如果你是信息安全爱好者&#xff0c;如果你想考一些证书来提升自己的能力&#xff0c;那么欢迎大家来我的 Discord 频道 Northern Bay。邀请链接在这里&#xff1a; https://discord.gg/9XvvuFq9Wb我拥有 OSCP&#xff0c;OSEP&#xff0c;OSWE&#xff0c;OSED&…...

每日一题29——山峰数组的顶部

符合下列属性的数组 arr 称为 山峰数组&#xff08;山脉数组&#xff09; &#xff1a; arr.length > 3 存在 i&#xff08;0 < i < arr.length - 1&#xff09;使得&#xff1a; arr[0] < arr[1] < ... arr[i-1] < arr[i] arr[i] > arr[i1] > ... &g…...

Linux- 系统随你玩之--好用到炸裂的系统级监控、诊断工具

文章目录1、前言2、lsof介绍2.1、问题来了&#xff1a; 所有用户都可以采用该命令吗&#xff1f;3、 服务器安装lsof3.1、安装3.2、检查安装是否正常。4、lsof 命令4.1、常用功能选项4.2、输出内容4.2.1 、FD和 TYPE列5、 lsof 命令实操常见用法6 、常用组合命令7、 结语1、前言…...

第十三节 继承

什么是继承&#xff1f; java中提供一个关键字extends&#xff0c;用这个关键字&#xff0c;我们可以让一个类和另一个类建立父子关系。 public class Student extends People{} student为子类&#xff08;派生类&#xff09;&#xff0c;people为父类&#xff08;基类或者超类…...

【优化】性能优化Springboot 项目配置内置Tomcat使用Http11AprProtocol(AIO)

Springboot 项目配置内置tomcat使用Http11AprProtocol(AIO) Windows版本 1.下载Springboot对应版本tomcat包 下载地址 Apache Tomcat - Apache Tomcat 9 Software Downloads 找到bin目录下 tcnative-1.dll 文件 2 放到jdk的bin目录下 Linux版本 在Springboot中内嵌的Tomcat默…...

SpringBoot之@ConfigurationProperties、@EnableConfigurationProperties

ConfigurationProperties 这个注解不仅可以为yml某个类注入还可以为第三方bean绑定属性 为yml某个类注入 只要将对应的yml类对象声明实体pojo并交给spring容器管理&#xff0c;再在类上使用ConfigurationProperties绑定对应的类名即可 涉及到两个知识点&#xff0c;这个类对…...

数组一次性删除多条数据

需求描述 最后提交时删除表格中的空行 实现方法 单行删除 - 并不是一次性删除 表格每行的最后设置删除按钮&#xff0c;点击时将当前行的索引传递给方法&#xff0c;splice 删除当前行。 <el-table :data"tableData" class"myTable" border>..…...

相机删除照片如何恢复?一键解决它

相机删除照片如何恢复&#xff1f;喜欢用相机拍照的人&#xff0c;总会在空闲时多拍几张&#xff0c;这使我们相机中会储存大量的、各种各样的照片。等到回家后&#xff0c;在进行删除&#xff0c;并选出比较好的照片。但也很容易就误删了一些好看的照片。碰到这种意外事&#…...

vue3搭建教程(基于webpack+create-vue+ element-plus)

前言使用vue脚手架搭建vuetswebpack项目搭建步骤&#xff1a;下载node 版本可以 12 或者14或者 16.0&#xff0c;此次使用的>16.0版本&#xff0c;vue-cli通过npm i -g vue/cli 升级到了 vue cli v5.0.8建目录&#xff0c;如&#xff08;vue3Study&#xff09;用IDE工具打开…...

代码随想录算法训练营第四十二天 | leetcode 1049. 最后一块石头的重量 II,494. 目标和,474.一和零

代码随想录算法训练营第四十二天 | leetcode 1049. 最后一块石头的重量 II&#xff0c;494. 目标和&#xff0c;474.一和零1049. 最后一块石头的重量 II494. 目标和474.一和零1049. 最后一块石头的重量 II 题目&#xff1a; 有一堆石头&#xff0c;每块石头的重量都是正整数。…...

Java8中Lambda表达式之Collection 的常见用法

背景 在java8中引入了Lambda表达式。其实&#xff0c;他就是一个匿名函数。我们经常会用到一些循环遍历&#xff0c;起始完全就可以通过Lambda来简化我们不必要的操作&#xff0c;下面我们来看一下Lambda常用的方法。 准备条件 DataBuilderprivate static class Person {priv…...

SpringCloud系列知识快速复习 -- part 2(Sentinel微服务保护,Seata分布式事务,Redis分布式缓存和多级缓存)

SpringCloud系列知识快速复习 -- part 2&#xff08;Sentinel微服务保护&#xff0c;Seata分布式事务&#xff0c;Redis分布式缓存和多级缓存Sentinel微服务保护什么是雪崩问题&#xff1f;解决方法服务保护技术对比流量控制簇点链路Sentinel流控模式流控效果热点参数限流隔离和…...

设置CentOS7的时间与网络同步

1.设置时区为北京时间 [rootlocalhost ~]# timedatectl set-timezone Asia/Shanghai 2.查看系统时间 [rootlocalhost ~]# timedatectl Local time: 四 2023-03-02 17:40:41 CST #系统时间 Universal time: 四 2023-03-02 09:40:41 UTC …...

java开发手册之编程规约

文章目录编程规约命名风格常量定义代码格式OOP规约集合处理并发处理控制语句注释规约其它编程规约 命名风格 1.代码中的命名均不能以下划线或者美元符号开始&#xff0c;也不能以下划线或者美元符号结束 例如&#xff1a;_name | name__ | name$ | $name2.代码中的命名严…...

Camera | 5.Linux v4l2架构(基于rk3568)

上一篇我们讲解了如何编写基于V4L2的应用程序编写&#xff0c;本文主要讲解内核中V4L2架构&#xff0c;以及一些最重要的结构体、注册函数。 厂家在实现自己的摄像头控制器驱动时&#xff0c;总体上都遵循这个架构来实现&#xff0c;但是不同厂家、不同型号的SoC&#xff0c;具…...

机房PDU如何挑选?

PDU PDU(Power Distribution Unit,电源分配单元),也就是我们常说的机柜用电源分配插座,PDU是为机柜式安装的电气设备提供电力分配而设计的产品,拥有不同的功能、安装方式和不同插位组合的多种系列规格,能为不同的电源环境提供适合的机架式电源分配解决方案。PDU的应用,…...

三电平逆变器实战:从SVPWM调制到中点平衡的硬核玩法

三电平逆变器 仿真 SVPWM调制 中点电位平衡控制 可选svpwm or spwm T型 I型NPC和ANPC&#xff08;拓扑都有可以选&#xff09; 包含三相逆变器参数设计&#xff0c;SVPWM&#xff0c;直流均压控制&#xff0c;双闭环控制说明文档 直流电压750V&#xff0c;输出交流电压220V&…...

MySQL慢查询开启与分析优化案例

一、前言1.1 什么是慢查询日志慢查询日志是MySQL提供的一种性能诊断工具&#xff0c;用于记录执行时间超过指定阈值的SQL语句。通过分析这些“慢SQL”&#xff0c;可以精准定位数据库性能瓶颈&#xff0c;优化索引、SQL写法或表结构。1.2 基础知识要求MySQL基础&#xff1a;熟悉…...

海康威视摄像头CVE-2017-7921漏洞复现:从零到一的实战指南(含解密工具下载)

海康威视摄像头CVE-2017-7921漏洞深度解析与实战复现 在物联网设备安全领域&#xff0c;监控摄像头的漏洞利用一直是渗透测试的热点话题。2017年曝光的海康威视摄像头认证绕过漏洞&#xff08;CVE-2017-7921&#xff09;因其影响范围广、利用难度低而备受关注。本文将带您深入理…...

倒立摆这玩意儿在控制领域算是个经典玩具,今天咱们用MATLAB整点有意思的——用LQR同时实现小车的平衡控制和外加摆杆起摆。先来点硬核的,直接上状态方程

MATLAB直线倒立摆一阶倒立摆LQR控制仿真&#xff0c;小车倒立摆起摆和平衡控制&#xff0c;附带参考文献% 系统参数&#xff08;单位全部国际标准制&#xff09; M 0.5; % 小车质量 m 0.2; % 摆杆质量 l 0.3; % 摆杆半长 g 9.81; % 重力加速度% 状态空间模型推导…...

最新!2026年3月全球大模型全景:国产登顶、百万上下文、智能体爆发,AI进入实用新纪元

&#x1f525;个人主页&#xff1a;北极的代码&#xff08;欢迎来访&#xff09; &#x1f3ac;作者简介&#xff1a;java后端学习者 ❄️个人专栏&#xff1a;苍穹外卖日记&#xff0c;SSM框架深入&#xff0c;JavaWeb ✨命运的结局尽可永在&#xff0c;不屈的挑战却不可须臾或…...

Z-Image-Turbo-辉夜巫女实战:基于卷积神经网络的特征引导图像风格迁移

Z-Image-Turbo-辉夜巫女实战&#xff1a;基于卷积神经网络的特征引导图像风格迁移 你是不是也遇到过这样的烦恼&#xff1f;看到一张特别有感觉的艺术画作&#xff0c;想把自己的照片也变成那种风格&#xff0c;结果用普通的滤镜一处理&#xff0c;要么颜色变得很奇怪&#xf…...

深入解析highway_env:强化学习自动驾驶环境的代码架构与实现

1. highway_env&#xff1a;强化学习自动驾驶的仿真利器 第一次接触highway_env时&#xff0c;我就被它的简洁设计惊艳到了。这个基于Python的开源项目&#xff0c;专门为自动驾驶强化学习研究提供了高度可配置的仿真环境。想象一下&#xff0c;你正在开发一个自动驾驶AI&#…...

Cloudflare图像转换免费了!手把手教你配置WebP/AVIF自动优化(附R2存储成本计算)

Cloudflare图像优化实战指南&#xff1a;从配置到成本控制的完整方案 Cloudflare近期宣布其图像转换功能全面免费开放&#xff0c;这一变化让中小型网站管理员和开发者能够零门槛享受现代图像格式带来的性能红利。作为全球领先的CDN服务商&#xff0c;Cloudflare的这一决策将We…...

MMA8452Q加速度传感器I²C驱动与嵌入式集成实战

1. MMA8452Q加速度传感器底层驱动技术解析1.1 器件特性与工程定位MMA8452Q是NXP&#xff08;现为恩智浦半导体&#xff09;推出的超低功耗、高精度三轴数字加速度传感器&#xff0c;采用331 mm QFN封装&#xff0c;专为便携式消费电子、可穿戴设备及工业状态监测等对尺寸、功耗…...

Ubuntu 22.04 LTS下Xenomai 3.3实时内核配置全攻略(附常见错误排查)

Ubuntu 22.04 LTS下Xenomai 3.3实时内核配置全攻略&#xff08;附常见错误排查&#xff09; 在工业自动化、机器人控制和实时数据处理领域&#xff0c;系统响应时间的确定性往往比绝对性能更重要。想象一下&#xff0c;当机械臂需要在毫秒级精度下完成轨迹规划&#xff0c;或者…...