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

文章目录

demo地址

demo工程地址

实现效果

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

引入

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

对应接口的响应就变成如下图所示:

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

基础类准备

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

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

java 复制代码
/**
 * @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之间
java 复制代码
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();
java 复制代码
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字段

java 复制代码
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异常,当发生不可预知的异常时,在此统一捕获,并打印错误日志
java 复制代码
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,并在其中手动抛出一个业务异常,如下:

java 复制代码
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时,返回对应的错误
  • 可以看到两者的数据结构都是完全一样的
相关推荐
IT学长编程10 小时前
计算机毕业设计 教师科研信息管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解
java·毕业设计·springboot·毕业论文·计算机毕业设计选题·计算机毕业设计开题报告·教师科研管理系统
孟诸15 小时前
计算机专业毕设-校园新闻网站
java·vue·毕业设计·springboot·课程设计
_院长大人_1 天前
SpringBoot 整合docker,执行容器服务
java·docker·springboot
我是小酒2 天前
掌握 Spring:从新手到高手的常见问题汇总
java·后端·spring·springboot
赚钱给孩子买茅台喝3 天前
智能BI项目第一期
java·人工智能·springboot·react
java1234_小锋3 天前
免费分享一套SpringBoot+Vue学生信息管理系统【论文+源码+SQL脚本】,帅呆了~~
java·springboot·java毕业设计·学生信息·java学生信息·springboot学生信息·vue学生信息
蓝染-惣右介3 天前
【若依RuoYi-Vue | 项目实战】帝可得后台管理系统(一)
java·后端·物联网·vue·springboot
张某布响丸辣4 天前
Nginx 负载均衡:优化网站性能与可扩展性的利器
java·运维·nginx·负载均衡·springboot
爱新觉罗15 天前
Mybatis-plus-Generator 3.5.5 自定义模板支持 (DTO/VO 等) 配置
mybatis·springboot
IT学长编程5 天前
计算机毕业设计 大学志愿填报系统的设计与实现 Java实战项目 附源码+文档+视频讲解
java·毕业设计·springboot·毕业论文·计算机毕业设计选题·计算机毕业设计开题报告·大学志愿填报系统