Spring MVC中异常处理

1.全局异常处理

1.1什么是全局异常处理器

全局异常处理器是SpringMVC框架中的一种异常处理机制,用于统一处理由控制器抛出的异常。

全局异常处理器可以帮助我们捕获和处理控制器中的异常,并且根据不同的异常类型进行不同的处理操作,从而保障应用的健壮性和稳定性。

当然,SpringMVC中有内置的异常处理对象,但是呈现的结果对于用户端不友好,所以实际项目我们一般会使用全局异常处理器处理异常。

如果仅仅在项目中写下以下代码,会导致异常处理呈现的结果用户端难以理解。

复制代码
@RestController
@RequestMapping("/v1/tests/")
public class TestController {

    @GetMapping("test")
    public JsonResult test(Integer id) {
        if (id < 0) {
            throw new IllegalArgumentException("id不能小于0");
        }
        return new JsonResult(200,"成功访问");
    }
}

如果我们能做到像一些企业级的异常应对措施,比如bilibili

当输入网址https://www.bilibili.com/id=111

这个地址根本不存在,结果返回一个"找不到页面"的页面。

这种处理效果是 前端+后端共同开发的效果,采用后端捕获异常 + 前端自定义错误页面渲染 的组合处理方式

前后端分离(REST 风格)

  • 用全局异常处理器统一返回结构化 JSON

  • 由前端(如 Vue、React)判断 status 或 code,显示漂亮的提示页面

{ "code": 404, "msg": "资源不存在", "data": null }

然后前端根据 code == 404 显示"找不到页面"的提示比如哔哩哔哩的效果。

1.2 全局异常处理器的配置

Spring MVC中的全局异常处理器可以通过以下方式进行配置:

  1. 创建 exception.GlobalExceptionHandler 类,并添加异常处理方法;

使用 @ControllerAdvice 注解 + ResponseBody注解 或者 @RestControllerAdvice 注解标注该类;

  1. 在异常处理方法上添加 @ExceptionHandler 注解,用于指定控制器中需要处理的异常类型。

@ControllerAdvice

@ControllerAdvice 本质上是一个带有 @Component 的注解,Spring 启动时会将它的类扫描进容器中。

它内部利用 AOP 和 HandlerExceptionResolver 原理,在 Controller 执行过程中如果抛出了异常,就会查找有没有全局的异常处理器处理它。

@ExceptionHandler

@Exception注解 指定:这个方法能处理哪种异常类型

它的参数是异常对象(如 IllegalArgumentException e),Spring 会自动注入

你可以根据异常信息生成 JSON 响应、记录日志等

我们来进行优化上面的代码,在TestController基础上加上GlobalHandlerException类

复制代码
@RestController
@RequestMapping("/v1/tests/")
public class TestController {

    @GetMapping("test")
    public JsonResult test(Integer id) {
        if (id < 0) {
            throw new IllegalArgumentException("id不能小于0");
        }
        return new JsonResult(200,"成功访问");
    }
}



@Slf4j
@RestControllerAdvice
public class GlobalHandlerException {

    @ExceptionHandler
    public String doHandlerIllegalArgumentExceptionException(IllegalArgumentException ex){
        log.error("ex : " + ex);
        return ex.getMessage();
    }
}

此时虽然没有哔哩哔哩网址那么华丽,但是总归是可以让用户清晰的知道不能传递id < 0这个限制了。

1.3 使用流程

1)创建全局异常处理器类

工程目录下创建 exception.GlobalExceptionHandler

@ControllerAdvice 注解

定义全局异常处理器,处理Controller中抛出的异常。

@RestControllerAdvice 注解

复合注解,是 @ControllerAdvice 注解和 @ResponseBody 注解的组合;

用于捕获Controller中抛出的异常并对异常进行统一的处理,还可以对返回的数据进行处理。

2)创建异常处理方法

在异常处理方法上添加 @ExceptionHandler 注解

@ExceptionHandler 注解

用于捕获Controller处理请求时抛出的异常,并进行统一的处理。

示例

复制代码
/**
    ex.getMessage()方法:用于捕获异常信息
*/
@ExceptionHandler
public JsonResult doHandleRuntimeException(RuntimeException ex){
    log.error("error is " + ex.getMessage());
    return new JsonResult(StatusCode.OPERATION_FAILED,ex.getMessage());
}

1.4 全局异常处理器示例

1)微博详情页异常抛出

复制代码
public JsonResult selectById(int id){
    if(id < 0) {
        throw new IllegalArgumentException("id值无效");
    }
    ... ...
}

2)全局异常处理

exception.GlobalExceptionHandler 类

复制代码
package cn.tedu.weibo.exception;

import cn.tedu.weibo.common.response.JsonResult;
import cn.tedu.weibo.common.response.StatusCode;
import lombok.Setter;
import lombok.extern.slf4j.Slf4j;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.annotation.*;

/**
 * RestControllerAdvice 是复合注解,描述的类型为一个全局异常处理对象类型,
 * 等价于:@ControllerAdvice+@ResponseBody
 * 当某个Controller方法中出现了异常,系统底层就会查找有没有定义全局异常处理对象。
 * 这个全局异常处理对象中有没有定义对应的异常处理方法,假如有就调用此方法处理异常。
 */

@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {
    /**
     *  @ExceptionHandler 描述的方法为一个异常处理方法,在此注解内部可以定义具体的异常处理
     *  类型(例如RuntimeException),此注解描述的方法需要定义一个异常类型的形式参数,
     *  通过这个参数接收具体的异常对象(也可以接收其异常类型对应的子类类型的异常)。
     */
    @ExceptionHandler
    public JsonResult doHandleRuntimeException(RuntimeException ex){
        log.error("error is " + ex.getMessage());
        return new JsonResult(StatusCode.OPERATION_FAILED,ex.getMessage());
    }
    
    /**
     * 假如用全局异常处理对象处理Controller类中出现的异常,全局异常处理对象会优先查找与Controller
     * 中相匹配的异常处理方法,假如没有,会查找对应异常的父类异常处理方法。
     */
    @ExceptionHandler
    public JsonResult doHandleRuntimeException(IllegalArgumentException ex){
        log.error("IllegalArgumentException is " + ex.getMessage());
        return new JsonResult(StatusCode.OPERATION_FAILED,ex.getMessage());
    }
}

3)重启工程测试

http://localhost:8080/v1/weibo/selectById?id=-1

2 关于Throwable

在开发实践中,通常会添加一个处理 Throwable 的方法,它将可以处理所有类型的异常,则不会再出现500错误!

Throwable 是 Java 所有异常(Exception)和错误(Error)的顶层父类

GlobalExceptionHandler中添加处理 Throwable 的方法

复制代码
@ExceptionHandler
public JsonResult handleThrowable(Throwable e) {
    return new JsonResult(8888, "程序运行过程中出现了Throwable");
}

这个方法千万不要随便加,不然后续出了异常就看不出来了,可以等到项目开发的差不多了,测试bug也都找全了,准备上线了再添加。