004-异常处理

2020/10/4 Spring Boot

之前的代码中,我们已经能够处理一些正常的情况的返回,对于一些异常的情况,我们还需要进行统一的捕获并且处理。

# 定义异常捕获

@ControllerAdvice
public class LaymanExceptionHandler {

    private static final String STATUS_PREFIX_4 = "4";
    private static final String STATUS_PREFIX_5 = "5";

    private static final String STATUS_400 = "400";
    private static final String STATUS_401 = "401";
    private static final String STATUS_403 = "403";
    private static final String STATUS_404 = "404";

    @ExceptionHandler(Exception.class)
    @ResponseBody
    public CommonResult exceptionHandler(HttpServletRequest request, HttpServletResponse response, Exception e) {

        CommonResult result = null;

        HashMap<Object, Object> error = MapUtil.newHashMap();

        error.put("detail", e.getMessage());
        error.put("path", request.getServletPath());

        result = CommonResult.failed(BResultCode.COMMON_B_ERROR, error);

        String status = String.valueOf(response.getStatus());
        if (StrUtil.startWith(status, STATUS_PREFIX_4)) {
            result = CommonResult.failed(AResultCode.COMMON_A_ERROR, error);
        } else if (StrUtil.startWith(status, STATUS_PREFIX_5)) {
            result = CommonResult.failed(BResultCode.COMMON_B_ERROR, error);
        }

        switch (status) {
            case STATUS_400:
                result = CommonResult.failed(AResultCode.COMMON_400_ERROR, error);
                break;
            case STATUS_401:
                result = CommonResult.failed(AResultCode.COMMON_401_ERROR, error);
                break;
            case STATUS_403:
                result = CommonResult.failed(AResultCode.COMMON_403_ERROR, error);
                break;
            case STATUS_404:
                result = CommonResult.failed(AResultCode.COMMON_404_ERROR, error);
                break;
            default:
                break;
        }

        if (e instanceof BizException) {
            BizException e2 = (BizException) e;
            result = CommonResult.failed(e2.getResultCode(), error);
        }

        return result;
    }

}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58

上面的代码就是我们共同拦截exception的全部内容,其中涉及到新增的部分我们一个一个解释一下。

首先@ControllerAdvice和@ExceptionHandler(Exception.class)所起到的作用就是将一场捕获并且返回,而@ResponseBody则是对我们的错误信息进行了json返回。 我们来看一下result的构成的话,就会发现CommonResult。failed()方法,我们可以看到下面的代码,其实就是构建了一个data等于null,error不为null的CommonResult对象。

@ApiModel(value = "共通返回值", description = "共通返回值")
@Data
@AllArgsConstructor
public class CommonResult<T> {
    private String code;
    private String message;
    private T data;
    private T error;

    /**
     * 成功返回结果
     *
     * @param data 获取的数据
     */
    public static <T> CommonResult<T> success(T data) {
        return new CommonResult<T>(ResultCode.OK.getCode(), ResultCode.OK.getMessage(), data, null);
    }


    /**
     * 失败返回结果
     *
     * @param error 获取的数据
     */
    public static <T> CommonResult<T> failed(IResultCode resultCode, T error) {
        return new CommonResult<T>(resultCode.getCode(), resultCode.getMessage(), null, error);
    }

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29

这里面那个resultCode的类型是IResultCode接口,我们事先了AResultCode,BResultCode,CResultCode,ResultCode四个枚举,

  • AResultCode 客户端错误
  • BResultCode 系统错误
  • CResultCode 第三方调用错误
  • ResultCode 正确信息

了解了CommonResult的返回值之后,我们再回过头来看exceptionHandler内的业务,首先构建一个error的map,放一些具体的信息,同时根据HTTPStatusCode来判断一些共通的异常类型。 比如400,401,403,404,500等。 而BizException,是我们自定义的业务错误信息,是我们抛出的信息内容。

@Data
@AllArgsConstructor
public class BizException extends RuntimeException {
    private static final long serialVersionUID = 1L;

    private IResultCode resultCode;

    private String message;

}
1
2
3
4
5
6
7
8
9
10

上述就是自定义的业务Exception的内容,需要手动抛出的时候,我们就可以throw new BizException()。

截止到现在,我们就可以实现了一个简单的全局异常抓取的工作,手动定义的的异常和意料外错误的遗产均能够捕获并按照我们的期望返回。

# 20201023修改

在随后的开发过程中,我们修改了一下ExceptionHandler的内容,将之前一个方法改变为多个方法,拦截不同类型Exception

@ControllerAdvice
@Slf4j
@Order(-100)
public class LaymanExceptionHandler {

    @ResponseBody
    @ResponseStatus(HttpStatus.BAD_REQUEST)
    @ExceptionHandler(value = MethodArgumentNotValidException.class)
    public CommonResult handleValidException(HttpServletRequest request, MethodArgumentNotValidException e) {

        return CommonResult.failed(AResultCode.COMMON_400_ERROR, createErrorDetail(request, e));
    }

    /**
     * 认证异常--认证失败(账号密码错误,账号被冻结,token过期等)
     */
    @ExceptionHandler(AuthException.class)
    @ResponseStatus(HttpStatus.UNAUTHORIZED)
    @ResponseBody
    public CommonResult unAuth(HttpServletRequest request, AuthException e) {

        return CommonResult.failed(AResultCode.COMMON_401_ERROR, createErrorDetail(request, e));
    }

    /**
     * 认证异常--没有访问权限
     */
    @ExceptionHandler(PermissionException.class)
    @ResponseStatus(HttpStatus.FORBIDDEN)
    @ResponseBody
    public CommonResult permissionException(HttpServletRequest request, PermissionException e) {
        return CommonResult.failed(AResultCode.COMMON_403_ERROR, createErrorDetail(request, e));
    }

    /**
     * 404
     */
    @ExceptionHandler(NoHandlerFoundException.class)
    @ResponseStatus(HttpStatus.NOT_FOUND)
    @ResponseBody
    public CommonResult notFoundException(HttpServletRequest request, NoHandlerFoundException e) {
        return CommonResult.failed(AResultCode.COMMON_404_ERROR, createErrorDetail(request, e));
    }

    /**
     * 拦截业务异常
     */
    @ExceptionHandler(BizException.class)
    @ResponseStatus(HttpStatus.CONFLICT)
    @ResponseBody
    public CommonResult businessException(HttpServletRequest request, BizException e) {
        return CommonResult.failed(AResultCode.COMMON_A_ERROR, createErrorDetail(request, e));
    }

    /**
     * 拦截未知的运行时异常
     */
    @ExceptionHandler(RuntimeException.class)
    @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
    @ResponseBody
    public CommonResult internalException(HttpServletRequest request, RuntimeException e) {
        return CommonResult.failed(BResultCode.COMMON_B_ERROR, createErrorDetail(request, e));
    }


    /**
     * 创建error详细
     *
     * @param request
     * @param e
     * @return map
     */
    private HashMap<Object, Object> createErrorDetail(HttpServletRequest request, Exception e) {

        StackTraceElement stackTraceElement = e.getStackTrace()[0];
        HashMap<Object, Object> error = MapUtil.newHashMap();
        error.put("path", request.getServletPath());
        error.put("className", stackTraceElement.getClassName());
        error.put("methodName", stackTraceElement.getMethodName());
        error.put("line", stackTraceElement.getLineNumber());
        error.put("detail", e.getMessage());

        return error;
    }

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85

我们可以看到,在获取不同类型的Exception上,我们分别采用不同方法来拦截。并且我们创建了一些固定类型的Exception来代表不同的业务,并且回复了相应的HTTPStatusCode。 同时在Exception返回的具体业务中,我们也增加了一些内容使得我们的error更为具体。

Last Updated: 2025/2/21 01:42:36