spring Boot 2.x | 自定义异常 统一返回值 全局异常处理

本文讲解如何在springboot项目中使用自定义异常,统一返回,全局异常处理

推荐访问我的个人网站,排版更好看: https://chenmingyu.top/springboot-exception/
系列文章: https://chenmingyu.top/tags/springboot/

新建一个springboot项目,引入web依赖,pom.xml完整依赖如下

<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <version>1.16.22</version>
    </dependency>
</dependencies>
自定义异常

在项目中有的地方,比如入参校验错误,安全校验失败等可能需要主动去抛异常,这时候就需要自己自定义一个异常。

/**
 * @auther: chenmingyu
 * @date: 2018/12/6 14:36
 * @description:
 */
@Data
public class MyException extends RuntimeException{

    private Integer code;

    private String msg;

    public MyException(Integer code, String msg) {
        this.code = code;
        this.msg = msg;
    }

    public MyException(String msg) {
        this.code = ResultTypeEnum.SERVICE_ERROR.getCode();
        this.msg = msg;
    }

    public MyException(ResultTypeEnum resultEnum) {
        this.code = resultEnum.getCode();
        this.msg = resultEnum.getMessage();
    }
}

自定义异常只要新建一个类继承RuntimeException类

统一返回值
定义统一返回类:Result

Result 类构造函数私有化,使用Result 类提供的static方法构造Result 实例返回,再定义一个响应状态的枚举ResultTypeEnum,同一返回状态码等信息。

/**
 * @auther: chenmingyu
 * @date: 2018/10/31 16:47
 * @description:
 */
public class Result<T> implements Serializable {
  
  private Integer code;
  private String msg;
  private T data;

  private Result() {
  }

  public Result(ResultTypeEnum type) {
    this.code = type.getCode();
    this.msg = type.getMessage();
  }

  public Result(ResultTypeEnum type, T data) {
    this.code = type.getCode();
    this.msg = type.getMessage();
    this.data = data;
  }

  public Result(ResultTypeEnum type, String content, T data) {
    this.code = type.getCode();
    this.msg = content;
    this.data = data;
  }

  public static Result success() {
    return new Result(ResultTypeEnum.SERVICE_SUCCESS);
  }

  public static <T> Result<T> success(T data) {
    return new Result(ResultTypeEnum.SERVICE_SUCCESS, data);
  }

  public static <T> Result<T> error(T data) {
    return new Result(ResultTypeEnum.SERVICE_ERROR, data);
  }

  public static  <T> Result<T> success(String content, T data) {
    return new Result(ResultTypeEnum.SERVICE_SUCCESS, content, data);
  }

  public static Result error() {
    return new Result(ResultTypeEnum.SERVICE_ERROR);
  }

  public static Result error(ResultTypeEnum typeEnum) {
    return new Result(typeEnum);
  }

  public static Result error(ResultTypeEnum typeEnum,String msg) {
    return new Result(typeEnum,msg);
  }

  public String getMsg() {
    return msg;
  }

  public void setMsg(String msg) {
    this.msg = msg;
  }

  public Integer getCode() {
    return code;
  }

  public void setCode(Integer code) {
    this.code = code;
  }

  public T getData() {
    return data;
  }

  public void setData(T data) {
    this.data = data;
  }
}
响应状态枚举:ResultTypeEnum
/**
 * @auther: chenmingyu
 * @date: 2018/10/31 16:47
 * @description:
 */
public enum ResultTypeEnum {

    SERVICE_SUCCESS(200,"成功"),
    PARAM_ERROR(40001,"入参异常"),
    SERVICE_ERROR(500,"服务异常");

    private Integer code;
    private String message;

    public int getCode() {
        return code;
    }

    public void setCode(int code) {
        this.code = code;
    }

    public String getMessage() {
        return message;
    }

    public void setMessage(String message) {
        this.message = message;
    }

    ResultTypeEnum(int code, String message) {
        this.code = code;
        this.message = message;
    }
}
怎么使用
/**
 * @auther: chenmingyu
 * @date: 2018/12/6 14:20
 * @description:
 */
@RestController
public class ExceptionController {

    /**
     * 成功返回值
     */
    @RequestMapping("/querySuccess")
    public Result<String> querySuccess(){
        return Result.success("我是数据");
    }

    /**
     * 错误返回值
     */
    @RequestMapping("/queryError")
    public Result<String> queryError(){
        return Result.error(ResultTypeEnum.SERVICE_ERROR);
    }
}

调用 http://localhost:8080/querySuccess
在这里插入图片描述

调用 http://localhost:8080/queryError

在这里插入图片描述

全局异常处理

在spring的项目中使用@ControllerAdvice注解来处理controller层报出的异常信息

//ControllerAdvice 注解源码
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface ControllerAdvice {
    @AliasFor("basePackages")
    String[] value() default {};

    @AliasFor("value")
    String[] basePackages() default {};

    Class<?>[] basePackageClasses() default {};

    Class<?>[] assignableTypes() default {};

    Class<? extends Annotation>[] annotations() default {};
}
全局异常处理类

@ExceptionHandler(value = Exception.class):处理Exception类型的异常

@ResponseStatus:响应的状态码

/**
 * @auther: chenmingyu
 * @date: 2018/12/6 10:57
 * @description:
 */
@ControllerAdvice
public class ExceptionHandle extends ResponseEntityExceptionHandler {

    /**
     * 只捕获 IllegalArgumentException 异常
     * @param request
     * @param e
     * @return
     */
    @ExceptionHandler(value = IllegalArgumentException.class)
    @ResponseBody
    @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    private Result illegalArgumentExceptionHandler(HttpServletRequest request, Exception e) {
        System.out.println("IllegalArgumentException 异常: " + e.getClass().getName());
        return Result.error(ResultTypeEnum.PARAM_ERROR,"请求地址:"+request.getRequestURI());
    }

    /**
     * 只捕获 MyException 异常
     * @param request
     * @param e
     * @return
     */
    @ExceptionHandler(value = MyException.class)
    @ResponseBody
    @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    private Result myExceptionHandler(HttpServletRequest request,MyException e) {
        System.out.println("自定义异常 :"+ e.getMsg());
        return  Result.error(e.getMsg()+"url:"+request.getRequestURI());
    }

    /**
     * 默认异常捕获,上面两个异常没走的话走我。
     * @param request
     * @param e
     * @return
     */
    @ExceptionHandler
    @ResponseBody
    @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    private Result exceptionHandler(HttpServletRequest request,Exception e) {
        System.out.println("异常 :"+ e.getClass().getName());
        request.getMethod();
        return Result.error(ResultTypeEnum.SERVICE_ERROR,"不符合上述异常的默认走我这");
    }
}
处理异常测试
/**
 * @auther: chenmingyu
 * @date: 2018/12/6 14:20
 * @description:
 */
@RestController
public class ExceptionController {

    /**
     * 抛出运行时异常
     */
    @RequestMapping("/throwRuntimeException")
    public void throwRuntimeException(){
        throw new RuntimeException();
    }

    /**
     * 抛出运行时异常
     */
    @RequestMapping("/throwIllegalArgumentException")
    public void throwIllegalArgumentException(){
        throw new IllegalArgumentException();
    }

    /**
     * 抛出自定义异常
     */
    @RequestMapping("/throwMyException")
    public void throwMyException(){
        throw new MyException("我是主动抛出来的");
    }
}

调用http://localhost:8080/throwIllegalArgumentException

在这里插入图片描述
调用http://localhost:8080/throwMyException

在这里插入图片描述

调用http://localhost:8080/throwRuntimeException

在这里插入图片描述

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,很高兴能回答您的问题。 首先,让我们了解一下 @ControllerAdvice 注解。它是 Spring MVC 框架提供的一种全局异常处理机制,可以让我们在一个地方集中处理所有 Controller 抛出的异常。 使用 @ControllerAdvice 注解的类可以包含多个 @ExceptionHandler 注解的方法,每个方法处理不同的异常类型。当 Controller 抛出异常Spring 会根据异常类型找到对应的 @ExceptionHandler 方法进行处理。 以下是一个简单的 @ControllerAdvice 示例: ```java @ControllerAdvice public class GlobalExceptionHandler { @ExceptionHandler(Exception.class) public ResponseEntity<String> handleException(Exception e) { return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("Something went wrong!"); } @ExceptionHandler(MyCustomException.class) public ResponseEntity<String> handleMyCustomException(MyCustomException e) { return ResponseEntity.status(HttpStatus.BAD_REQUEST).body(e.getMessage()); } } ``` 在上面的例子中,我们定义了两个 @ExceptionHandler 方法,分别处理 Exception 和 MyCustomException 类型的异常。 如果您想要更细粒度地处理异常,可以在 @ExceptionHandler 注解中指定具体的异常类型,如下所示: ```java @ExceptionHandler({IOException.class, SQLException.class}) public void handleIOExceptionOrSQLException() { // ... } ``` 另外,@ControllerAdvice 还支持自定义返回值类型和异常解析器,具体可以参考 Spring 官方文档。 最后,关于统一处理异常,除了使用 @ControllerAdvice,我们还可以使用 Spring Boot 提供的 ErrorController 接口或实现自己的异常处理器。 希望能对您有所帮助。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值