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

Spring异常处理-@ExceptionHandler-@ControllerAdvice-全局异常处理

文章目录

  • @ResponseBody
  • @ControllerAdvice
  • 最终的异常处理方式

异常的处理分两类
编程式处理:也就是我们的try-catch
声明式处理:使用注解处理
在这里插入图片描述

@ResponseBody


/*** 测试声明式异常处理*/
@RestController
public class HelloController {//编程式的异常处理;//如果大量业务都需要加异常处理代码的话,会很麻烦
//        try {
//            //执行业务
//
//        }catch (Exception e){
//            return R.error(100,"执行异常");
//        }@GetMapping("/hello")public R hello(@RequestParam(value = "i",defaultValue = "0") Integer i) throws FileNotFoundException {int j = 10 / i;//        FileInputStream inputStream = new FileInputStream("D:\\123.txt");String s = null;s.length();return R.ok(j);}/*** 1、如果Controller本类出现异常,会自动在本类中找有没有@ExceptionHandler标注的方法,*    如果有,执行这个方法,它的返回值,就是客户端收到的结果*  如果发生异常,多个都能处理,就精确优先* @return*/@ResponseBody@ExceptionHandler(ArithmeticException.class)public R handleArithmeticException(ArithmeticException ex){System.out.println("【本类】 - ArithmeticException 异常处理");return R.error(100,"执行异常:" + ex.getMessage());}@ExceptionHandler(FileNotFoundException.class)public R handleException(FileNotFoundException ex){System.out.println("【本类】 - FileNotFoundException 异常处理");return R.error(300,"文件未找到异常:" + ex.getMessage());}@ExceptionHandler(Throwable.class)public R handleException02(Throwable ex){System.out.println("【本类】 - Throwable 异常处理");return R.error(500,"其他异常:" + ex.getMessage());}}

@ExceptionHandler只能处理本类的
所以其他类的报错怎么办呢?
使用@ControllerAdvice

@ControllerAdvice

// 全局异常处理器
//@ResponseBody   // 结果还是以JSON的形式写出去
//@ControllerAdvice //告诉SpringMVC,这个组件是专门负责进行全局异常处理的
@RestControllerAdvice   // 合成注解
public class GlobalExceptionHandler {/*** 如果出现了异常:本类和全局都不能处理,* SpringBoot底层对SpringMVC有兜底处理机制;自适应处理(浏览器响应页面、移动端会响应json)* 最佳实践:我们编写全局异常处理器,处理所有异常* <p>* 前端关心异常状态,后端正确业务流程。* 推荐:后端只编写正确的业务逻辑,如果出现业务问题,后端通过抛异常的方式提前中断业务逻辑。前端感知异常;* <p>* 异常处理:* 1、** @param e* @return*/@ExceptionHandler(ArithmeticException.class)public R error(ArithmeticException e) {System.out.println("【全局】 - ArithmeticException 处理");return R.error(500, e.getMessage());}@ExceptionHandler(BizException.class)public R handleBizException(BizException e) {Integer code = e.getCode();String msg = e.getMsg();return R.error(code, msg);}@ExceptionHandler(value = MethodArgumentNotValidException.class)public R methodArgumentNotValidException(MethodArgumentNotValidException ex) {//1、result 中封装了所有错误信息BindingResult result = ex.getBindingResult();List<FieldError> errors = result.getFieldErrors();Map<String, String> map = new HashMap<>();for (FieldError error : errors) {String field = error.getField();String message = error.getDefaultMessage();map.put(field, message);}return R.error(500, "参数错误", map);}// 最终的兜底@ExceptionHandler(Throwable.class)public R error(Throwable e) {System.out.println("【全局】 - Exception处理" + e.getClass());return R.error(500, e.getMessage());}}

异常处理优先级

  • 本类 > 全局
  • 精确 > 模糊
    如果出现了异常,本类和全局都不能处理,SpringMVC会兜底处理机制: 自适应处理(什么样的客户端返回什么,要是浏览器就返回一个错误页面,要是客户端,比如Postman,返回json)
    实际上做项目的时候:我们编写全局异常处理器,处理所有异常

最终的异常处理方式

前端关心异常状态
后端正确业务流程
推荐:后端只编写正确的业务逻辑,如果出现业务问题,后端通过抛异常的方式提前中断业务逻辑。前端感知异常;
定义一个业务异常


/*** 业务异常* 大型系统出现以下异常:异常处理文档,固化* 1、订单  1xxxx*      10001 订单已关闭*      10002 订单不存在*      10003 订单超时*      .....* 2、商品  2xxxx*       20001 商品已下架*       20002 商品已售完*       20003 商品库存不足*       ......* 3、用户*      30001 用户已注册*      30002 用户已登录*      30003 用户已注销*      30004 用户已过期** 4、支付*      40001 支付失败*      40002 余额不足*      40003 支付渠道异常*      40004 支付超时** 5、物流*      50001 物流状态错误*      50002 新疆得加钱*      50003 物流异常*      50004 物流超时** 异常处理的最终方式:* 1、必须有业务异常类:BizException* 2、必须有异常枚举类:BizExceptionEnume  列举项目中每个模块将会出现的所有异常情况* 3、编写业务代码的时候,只需要编写正确逻辑,如果出现预期的问题,需要以抛异常的方式中断逻辑并通知上层。* 4、全局异常处理器:GlobalExceptionHandler;  处理所有异常,返回给前端约定的json数据与错误码*/@Data
public class BizException extends RuntimeException {private Integer code; //业务异常码private String msg; //业务异常信息public BizException(Integer code, String message) {super(message);this.code = code;this.msg = message;}public BizException(BizExceptionEnume exceptionEnume) {super(exceptionEnume.getMsg());this.code = exceptionEnume.getCode();this.msg = exceptionEnume.getMsg();}
}

为了便于管理,我们把所有的异常码和异常信息写一个枚举类。

public enum BizExceptionEnume {// ORDER_xxx:订单模块相关异常// PRODUCT_xxx:商品模块相关异常// 动态扩充.....ORDER_CLOSED(10001, "订单已关闭"),ORDER_NOT_EXIST(10002, "订单不存在"),ORDER_TIMEOUT(10003, "订单超时"),PRODUCT_STOCK_NOT_ENOUGH(20003, "库存不足"),PRODUCT_HAS_SOLD(20002, "商品已售完"),PRODUCT_HAS_CLOSED(20001, "商品已下架");@Getterprivate Integer code;@Getterprivate String msg;private BizExceptionEnume(Integer code, String msg) {this.code = code;this.msg = msg;}}

然后在我们的全局异常处理器中处理我们的业务异常

 @ExceptionHandler(BizException.class)public R handleBizException(BizException e) {Integer code = e.getCode();String msg = e.getMsg();return R.error(code, msg);}

在业务中就可以这样用了。

@Overridepublic void updateEmp(Employee employee) {//防null处理。考虑到service是被controller调用的;//controller层传过来的employee 的某些属性可能为null,所以先处理一下//怎么处理?Long id = employee.getId();if(id == null){ //页面没有带id//中断的业务的时候,必须让上层及以上的链路知道中断原因。推荐抛出业务异常throw new BizException(BizExceptionEnume.ORDER_CLOSED);}……

http://www.mrgr.cn/news/36675.html

相关文章:

  • AI应用开发中智能体编排应用是什么?
  • AI绘画美女指令大全,5个技巧让你的画作惊艳四座,美得令人窒息
  • Javascript编译原理
  • 浅谈C++之指针
  • 力扣(LeetCode)每日一题 2535. 数组元素和与数字和的绝对差
  • svn add代码时,忽略node_modules
  • 江科大笔记——新建工程
  • 代码随想录算法训练营第55天 | 寻找存在的路径
  • 大数据毕业设计选题推荐-租房数据分析系统-Hive-Hadoop-Spark
  • 浅谈C++之线程管理
  • 神经网络(五):U2Net图像分割网络
  • CSP-J 2024 入门级 第一轮(初赛) 阅读程序(1)
  • 【高阶数据结构】平衡二叉树(AVL)的插入(4种旋转方法+精美图解+完整代码)
  • PHP实现的纵横四海程序
  • 神经网络(四):UNet图像分割网络
  • 你了解文档透明加密系统吗?介绍7款顶尖文档透明加密软件,热门推荐!
  • Linux系统sersync数据实时同步
  • How to batch wise grad
  • Go语言匿名字段使用与注意事项
  • 跨地域协作新篇章:异地传输文件的最优方案