目录
依赖
一、全局异常处理
1、系统异常
2、业务异常
(1)、接口配置
(2)、服务配置
二、自定义异常
依赖
        <!-- validator,用于接口参数校验 -->
        <dependency>
            <groupId>org.hibernate.validator</groupId>
            <artifactId>hibernate-validator</artifactId>
        </dependency>一、全局异常处理
1、系统异常
在common目录中新建一个BaseException.class,用于接受全局的异常,一定要附上@RestControllerAdvice,才能被识别。
/**
 * 全局异常处理器
 */
@RestControllerAdvice
public class BaseException {
}对于参数格式有问题、代码运行错误等不可控因素的系统报错,默认的效果是这样:

而以下是系统异常的封装处理,返回我们规范的报错信息,
@ExceptionHandler是针对不同的报错类
@RequestLogAnnotation是我的aop日志处理,用于保存mysql
/**
 * 全局异常处理器
 */
@RestControllerAdvice
public class BaseException {
    /**
     * 系统异常
     * @param e 异常信息
     * @return Result
     */
    @ExceptionHandler(value = {Exception.class})
    @RequestLogAnnotation
    @ResponseBody
    public Result exceptionHandler(Exception e){
        return Result.error(500,"系统异常错误,请联系系统管理员处理!");
    }
}他的结果就是这样的:

2、业务异常
(1)、接口配置
我这里以post接口@RequestBody为例,接口如下,给dto对象前加上@Validated,
     /**
         * 登录
         * @param dto 暂存输入参数dto对象
         * @return
         */
        @RequestLogAnnotation(value = "用户登录")
        @PostMapping("/login")
        public Result login(@RequestBody @Validated LoginDto dto) {
                        // 进行对比
                        Login res = loginService.userLogin(dto);
                        // 判断结果是否返回该账号信息
                        if (res != null) {   
                                return Result.success(200, "登录成功", data);
                        } else {
                                return Result.error(501, "登陆失败");
                        }
        }然后,在实体类中给对应参数,加上@NotNull,其中的【(message = "用户账号不能为空")】这一段是在参数未空时报错返回的信息,用于返回给前端,但需要后台处理。
/**
 * 接受登录请求的参数
 */
@Data
public class LoginDto {
    @NotNull(message = "用户账号不能为空")
    private String id;
    @NotNull(message = "用户密码不能为空")
    private String password;
}(2)、服务配置
最后在前面的全局异常类中添加如下代码:
/**
     * 请求参数缺失异常处理
     * @param ex
     * @return
     */
    @ExceptionHandler(value = {MethodArgumentNotValidException.class})
    @ResponseStatus(HttpStatus.OK)
    @RequestLogAnnotation
    @ResponseBody
    public Result handleMethodArgumentNotValidException(MethodArgumentNotValidException ex) {
        Result result = new Result();
        BindingResult bindingResult = ex.getBindingResult();
        StringBuilder sb = new StringBuilder("校验失败:");
        for (FieldError fieldError : bindingResult.getFieldErrors()) {
            sb.append(fieldError.getDefaultMessage()).append(", ");
        }
        String msg = sb.toString();
        result.setCode(500);
        result.setMsg(msg);
        return result;
    }这时候请求里因为参数缺失,就会通过这个方法报错:

二、自定义异常
新建一个自定义异常类ServiceException.class,继承RuntimeException,如下(这里为了简化,只留一个getter方法和1个构造函数):
public class ServiceException extends RuntimeException{
    private Integer code;
    public Integer getCode() {
        return code;
    }
    public ServiceException (Integer code, String msg) {
        super(msg);
        this.code = code;
    }
}然后在全局异常类中添加如下代码(和系统异常基本一致,只需要更换@ExceptionHandler中的对应类即可):
    /**
     * 如果抛出的的是ServiceException,则调用该方法
     * @param e 服务异常
     * @return Result
     */
    @ExceptionHandler(ServiceException.class)
    @RequestLogAnnotation
    @ResponseBody
    public Result serviceExceptionHandle(ServiceException e){
        return Result.success(e.getCode(), e.getMessage());
    }这时候就可以在接口中使用try和catch方法进行判断异常使用了,如下:
         // sql执行判断
        try{
            res = getOne(queryWrapper);
        } catch (Exception e) {    
            throw new ServiceException(500, e);
        }


















