springboot2.x使用@RestControllerAdvice实现通用异常捕获

文章目录

    • demo地址
    • 实现效果
    • 引入
    • 基础类准备
      • 1.通用枚举与错误状态枚举
      • 2.定义通用返回结果
      • 3.自定义业务异常
    • 统一异常捕获
    • 测试

demo地址

demo工程地址

实现效果

  • 当我们输入1时,正常的返回通用的响应结果
  • 当我们输入2时,抛出异常,被捕获然后返回通用的结果
  • 可以看到两者的数据结构都是完全一样的

java通用异常捕获效果展示

引入

很多时候,当一个javaweb项目在运行的过程中出现一些不可预值的错误时会抛出异常,例如下方我们在接口直接抛出一个运行时异常:

请添加图片描述
对应接口的响应就变成如下图所示:

请添加图片描述
这时的返回结果往往和我们与前端约定好的响应结果的数据结构出入很大,而且也不方便我们去排查错误,并且对用户体验也不好,这时就需要我们配置一个统一的异常捕获,对各种异常进行统一数据结构的返回,方便前端展示对应的错误,我们也可以在捕获到异常后进行对应的日志记录,方便后续排查。

基础类准备

1.通用枚举与错误状态枚举

我们首先需要和前端沟通好统一的返回结果的数据结构和一些常见的错误状态,这里我们使用一个错误状态枚举来展示业务内的一些报错,为了枚举更好的拓展性,这里我们先封装一个基础枚举类,如下:

  • 封装通用枚举的好处,可以参考我的这几个视频讲解:
  • 封装通用枚举1
  • 封装通用枚举2(封装选项生成)
  • 封装通用枚举3(按类型获取枚举选项接口)
/**
 * @Author: lzp
 * @description: 通用枚举
 * @Date: 2022/9/24
 */
public interface BaseEnum<P> {

	/**
	 * 获取标题
	 */
	String getTitle();

	/**
	 * 获取值
	 */
	P getValue();

	/**
	 * 通过value获取枚举对象
	 * 2022-12-09日使用泛型优化此方法
	 *
	 * @param enumClass 枚举的类对象
	 * @param value     值
	 * @return
	 */
	static <T extends BaseEnum> T valueOf(Class<T> enumClass, Object value) {
		if (value == null) {
			return null;
		}
		T[] enumConstants = enumClass.getEnumConstants();
		if (enumConstants == null) {
			return null;
		}
		for (T enumConstant : enumConstants) {
			if (value.equals(enumConstant.getValue())) {
				return enumConstant;
			}
		}
		return null;
	}


}

接着,我们实现通用枚举接口,定义通用状态码枚举如下:

  • 这样可以把我们系统中可预知的异常全部定义在错误码枚举中
  • 我们可以给对应业务模块的错误码命名为指定范围内,例如用户相关的错误咱们控制在 10001 ~ 10100之间
import lombok.Getter;
import online.longzipeng.mywebdemo.enums.BaseEnum;

/**
 * @Author: lzp
 * @Date:2023/10/30
 * @description: 通用错误状态码
 */
@Getter
public enum ErrorCodeEnum implements BaseEnum<Integer> {

	// 通用错误状态码
	SUCCESS(0, "成功"),
	ERROR(-1, "失败"),
    
	// 用户相关异常  10001 ~ 10100
	USER_LOGIN_ERROR(1001,"账号或密码错误!"),
	;

	public final Integer value;
	public final String title;

	ErrorCodeEnum(Integer value, String title) {
		this.value = value;
		this.title = title;
	}

}

2.定义通用返回结果

我们与前端约定好通用的返回结果的数据结构,例如都带有code,msg,data,分别表示该接口的响应状态码,错误消息,返回数据:

  • 为了通用防止任何数据类型,这里我们使用泛型来指定响应的data
  • 为了方便通用的返回,例如一般修改接口不需要返回数据,此时在咱们就可以指定调用静态方法 Result.success();
package online.longzipeng.mywebdemo.commen;

import lombok.Data;
import online.longzipeng.mywebdemo.commen.exception.ErrorCodeEnum;

import java.io.Serializable;

/**
 * 通用响应
 *
 * @author lzp
 */
@Data
public class Result<T> implements Serializable {
	private static final long serialVersionUID = 1L;
	/**
	 * 编码:0表示成功,其他值表示失败
	 */
	private int code = ErrorCodeEnum.SUCCESS.value;
	/**
	 * 消息内容
	 */
	private String msg = ErrorCodeEnum.SUCCESS.title;
	/**
	 * 响应数据
	 */
	private T data;

	public static <T> Result<T> error() {
		return generate(ErrorCodeEnum.ERROR.value, ErrorCodeEnum.ERROR.getTitle());
	}

	/**
	 * 快速生成返回结果
	 * @param code 状态码
	 * @param msg 对应消息内容
	 */
	public static <T> Result<T> generate(int code, String msg) {
		Result<T> result = new Result();
		result.setCode(code);
		result.setMsg(msg);
		return result;
	}

	public static <T> Result<T> success() {
		return new Result<>();
	}

	public static <T> Result<T> success(T data) {
		Result<T> result = new Result<>();
		result.setData(data);
		return result;
	}

}

3.自定义业务异常

因为运行时异常是一种特殊的异常,不需要我们显示的抛出和try catch处理,所以很适合用于做我们的自定义业务异常,然后我们希望业务出错了也统一返回Result的数据结构,所以咱们自定义的异常也需要包含 code和msg字段

package online.longzipeng.mywebdemo.commen.exception;

import lombok.Data;

/**
 * 通用异常处理
 */
@Data
public class ServiceException extends RuntimeException {
	private static final long serialVersionUID = 1L;

	/**
	 * 错误码
	 */
	private int code;

	/**
	 * 错误信息
	 */
	private String msg;

	public ServiceException(int code) {
		this.code = code;
	}

	public ServiceException(int code, String msg) {
		this.code = code;
	}

	public ServiceException(String msg) {
		super(msg);
		this.code = ErrorCodeEnum.ERROR.value;
		this.msg = msg;
	}

	/**
	 * 使用通用错误枚举快速创建异常
	 */
	public ServiceException(ErrorCodeEnum errorCodeEnum) {
		this.code = errorCodeEnum.getValue();
		this.msg =errorCodeEnum.getTitle();
	}
}

统一异常捕获

在springboot2.x中,咱们可以通过@ControllerAdvice注解与@ExceptionHandler注解来实现统一的异常捕获与处理,为了方便返回结果,这里我们使用@RestControllerAdvice注解返回JSON格式的响应,用于快速构建RESTful风格的程序。

如下:

  • 这里我们直接捕获到了自定义的业务异常,从中取出结果并返回Result对象
  • 捕获Exception异常,当发生不可预知的异常时,在此统一捕获,并打印错误日志
package online.longzipeng.mywebdemo.commen.exception;

import online.longzipeng.mywebdemo.commen.Result;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;

/**
 * 通用异常处理器
 */
@RestControllerAdvice
public class ServiceExceptionHandler {
	private Logger logger = LoggerFactory.getLogger(getClass());

	/**
	 * 处理自定义异常
	 */
	@ExceptionHandler(ServiceException.class)
	public Result handleRenException(ServiceException e) {
		return Result.generate(e.getCode(),e.getMsg());
	}

	/**
	 * 处理未知异常
	 */
	@ExceptionHandler(Exception.class)
	public Result handleException(Exception e) {
		logger.error(e.getMessage(), e);
		return Result.error();
	}
}

测试

我们创建一个用于测试的controller,并在其中手动抛出一个业务异常,如下:

package online.longzipeng.mywebdemo.controller;

import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import io.swagger.annotations.ApiParam;
import online.longzipeng.mywebdemo.commen.Result;
import online.longzipeng.mywebdemo.commen.exception.ErrorCodeEnum;
import online.longzipeng.mywebdemo.commen.exception.ServiceException;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

/**
 * @Author: lzp
 * @description:
 * @Date: 2023/10/30
 */
@RestController
@RequestMapping("/test")
@Api(tags = "测试接口")
public class TestController {

	@GetMapping("/error")
	@ApiOperation("测试异常抛出")
	public Result<String> testError(@RequestParam @ApiParam("1 正常 2抛出错误") Integer type) {
		if (type == 2) {
			throw new ServiceException(ErrorCodeEnum.USER_LOGIN_ERROR);
		}
		return Result.success("你好呀~");
	}

}

显示结果如下:

  • 当我们输入1时,正常的返回通用的响应结果
  • 当我们输入2时,返回对应的错误
  • 可以看到两者的数据结构都是完全一样的

java通用异常捕获效果展示

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

macOS 安装brew

参考链接&#xff1a; https://mirrors4.tuna.tsinghua.edu.cn/help/homebrew/ https://www.yii666.com/blog/429332.html 安装中科大源的&#xff1a; https://zhuanlan.zhihu.com/p/470873649

《TCP/IP详解 卷一:协议》第5章的IPv4数据报的Checksum(校验和)字段的计算(这里才能解开你的困惑)

首先&#xff0c;我当你看过书&#xff0c;但是比较懵。 1&#xff0c;实例说明Checksum(校验和)的计算步骤 直奔主题&#xff0c;分析一下这个Checksum&#xff08;校验和&#xff09;怎么算出来的。 先用Wireshark随便抓一个UDP或TCP包分析一下。 如上面&#xff0c;我们得…

Python数据分析:在职场中的竞争优势

前言 在职场中&#xff0c;技能的重要性是不言而喻的。越来越多的职位要求员工具备数据分析能力&#xff0c;而Python作为一种强大的数据分析工具&#xff0c;正在成为职场中的“利器”。然而&#xff0c;尽管Python数据分析提供了巨大的优势&#xff0c;许多人依然未能掌握这…

团队表 -多级团队设计

团队表 -多级团队设计 user_team团队表 &#xff0c;如果存在子团队 1.我们可以通过每一个团队字段加一个parentid &#xff08;相当于一对多的关系&#xff09; 2.还可以设置一个字段CodingNum,比如这样: //系统为了管理查询团队自动生成的有序编号 可以使用3位数代表一个…

06.Oracle数据备份与恢复

Oracle数据备份与恢复 一、通过RMAN方式备份二、使用emp/imp和expdb/impdb工具进行备份和恢复三、使用Data guard进行备份与恢复 一、通过RMAN方式备份 通过 RMAN&#xff08;Oracle 数据库备份和恢复管理器&#xff09;方式备份 Oracle 数据库&#xff0c;可以使用以下步骤&a…

【漏洞复现】74cms任意文件读取

漏洞描述 74CMS 是一款国内用的比较多招聘网站管理系统&#xff08;Job Board CMS&#xff09;&#xff0c;专注于招聘和人力资源领域的网站建设&#xff0c;存在任意文件读取漏洞 免责声明 技术文章仅供参考&#xff0c;任何个人和组织使用网络应当遵守宪法法律&#xff0c…

Temp directory ‘C:\WINDOWS\TEMP‘ does not exist

问题描述 解决方法 管理员权限问题&#xff0c;进入temp文件夹更改访问权限即可。 点击 temp文件夹 属性 -> 安全 -> 高级 -> 更改主体Users权限 给读取和写入权限 参考博客 开发springboot项目时无法启动Temp directory ‘C: \WINDOWS\TEMP‘ does not exist

K-edge 和逃逸问题

一 k-eage基本概念 1 k-edge概念 K-edge称为K边, 其物理意义是高原子序数物质原子内部K层自由电子, 易与特定能量下X射线光子发生光电吸收作用, 导致对该能量的X射线光子吸收特别大。 而K-edge特性表现为X射线与物质发生相互作用时, 其衰减系数随着能量的增加而逐渐减小, 但在…

Postgresql批量按照顺序更新某一个字段

如批量更新采购订单行sequence字段&#xff0c;按照订单行id的顺序赋值1&#xff0c;2&#xff0c;3&#xff0c;4...&#xff1a; UPDATE purchase_order_line_copy1 SET sequence subquery.new_sequence FROM (SELECT id, ROW_NUMBER() OVER (ORDER BY id) AS new_sequence…

leetcode 117

leetcode 117 代码 #include <iostream>// Definition for a Node. class Node { public:int val;Node* left;Node* right;Node* next;Node() : val(0), left(NULL), right(NULL), next(NULL) {}Node(int _val) : val(_val), left(NULL), right(NULL), next(NULL) {}No…

python自动化测试模板

1:准备html模版 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 <!DOCTYPE html> <html> <head> <meta charset"utf-8"> <title>接口自动化…

天空卫士为集度智能汽车系上“安全带”

10月27日&#xff0c;集度汽车在北京正式发布了旗下首款量产车型——极越 01 SUV。极越 01 SUV 是一款集科技、智能、美学于一身的纯电动中大型SUV&#xff0c;号称全球首款“AI 汽车机器人”。作为集度的合作伙伴&#xff0c;天空卫士第一时间送上祝福&#xff0c;祝愿极越大卖…

基于LDA主题+协同过滤+矩阵分解算法的智能电影推荐系统——机器学习算法应用(含python、JavaScript工程源码)+MovieLens数据集(二)

目录 前言总体设计系统整体结构图系统流程图 运行环境模块实现1. 数据爬取及处理 相关其它博客工程源代码下载其它资料下载 前言 前段时间&#xff0c;博主分享过关于一篇使用协同过滤算法进行智能电影推荐系统的博文《基于TensorFlowCNN协同过滤算法的智能电影推荐系统——深…

centos7-lamp

目录 一、安装 1.关闭防火墙关闭selinux 2.安装apache 3.配置主页 二、部署mariadb&#xff08;mysql&#xff09; 1.用yum安装 2.启动数据库 3.看下端口是否listen 4登录mysql 5.修改下密码 三、安装php 1.安装依赖包 2.安装php解释器和php连接mysql驱动 3.配置…

Pure-Pursuit 跟踪双移线 Gazebo 仿真

Pure-Pursuit 跟踪双移线 Gazebo 仿真 主要参考学习下面的博客和开源项目 自动驾驶规划控制&#xff08;&#xff21;*、pure pursuit、LQR算法&#xff0c;使用c在ubuntu和ros环境下实现&#xff09; https://github.com/NeXTzhao/planning Pure-Pursuit 的理论基础见今年六月…

大模型在代码缺陷检测领域的应用实践

作者 | 小新、车厘子 导读 静态代码扫描(SA)能快速识别代码缺陷&#xff0c;如空指针访问、数组越界等&#xff0c;以较高ROI保障质量及提升交付效率。当前扫描能力主要依赖人工经验生成规则&#xff0c;泛化能力弱且迭代滞后&#xff0c;导致漏出。本文提出基于代码知识图谱解…

【python基础】python切片—如何理解[-1:],[:-1],[::-1]的用法

文章目录 前言一、基本语法二、切片1.a[i:j]2.a[i:j:k] 总结&#xff1a;[-1] [:-1] [::-1] [n::-1] 前言 在python中&#xff0c;序列是python最基本的数据结构&#xff0c;包括有string&#xff0c;list&#xff0c;tuple等数据类型&#xff0c;切片对序列型对象的一种索引方…

Spring Boot Actuator 漏洞利用

文章目录 前言敏感信息泄露env 泄露配置信息trace 泄露用户请求信息mappings 泄露路由信息heapdump泄露堆栈信息 前言 spring对应两个版本&#xff0c;分别是Spring Boot 2.x和Spring Boot 1.x&#xff0c;因此后面漏洞利用的payload也会有所不同 敏感信息泄露 env 泄露配置信…

【音视频 | Ogg】RFC3533 :Ogg封装格式版本 0(The Ogg Encapsulation Format Version 0)

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; &#x1f923;本文内容&#x1f923;&a…

Linux越学越头疼,我要怎么办?

最近&#xff0c;听到一些同学说&#xff0c;“Linux越学越头疼”。其实这句话&#xff0c;在我之前刚接触Linux的时候&#xff0c;也是深有感触。Linux越学越不明所以。最后干脆放弃学习&#xff0c;转而学习其他东西。 其实大家在初学Linux的时候&#xff0c; 有这个感受&am…