【Spring Boot】返回统一的JSON标准格式

近年来,随着移动互联网的发展,各种类型的客户端层出不穷。如果不统一数据接口,则会造成冗余编码,增加成本。RESTful 风格的 API 正适合通过一套统一的接口为 PC、手机 APP 等设备提供数据服务。

为了保障前后端数据交换的顺畅,建议规范数据的返回,并采用固定的数据格式封装。如:
返回成功信息的 JSON 格式如下

{
  "code": 200,
  "msg": "操作成功",
  "data": "hello jenkins"
}

返回异常信息的 JSON 格式如下

{
  "code": 500,
  "msg": "系统异常,请稍后重试:/by zero"
}

实现步骤如下

1. 自定义状态码枚举类(不是 HTTP 状态码)

@AllArgsConstructor
@Getter
public enum StatusCodeEnum {
    SC200 (200, "操作成功"),
    SC999 (999, "操作失败"),
    SC401 (401, "匿名用户访问权限资源时的异常"),
    SC403 (403, "无访问权限,请联系管理员授予权限"),
    SC404 (404, "请求的资源不存在"),
    SC500 (500, "系统异常,请稍后重试"),
    //... 略
    private final Integer code;
    private final String msg;
}

2. 封装返回结果

类名可任意取,可根据个人习惯或项目编码规约选择。比如 Result、ResultXxx、R、ApiResult、Response、ResponseXxx 等等,这里命名为 ApiResult

@Data
public class ApiResult<T> implements Serializable {
    private Integer code;
    private String msg;
    private T data;

    public static <T> ApiResult<T> success (T data) {
        return ApiResult.success (StatusCodeEnum.SC200.getMsg (), data);
    }

    public static <T> ApiResult<T> success (String msg, T data) {
        ApiResult<T> apiResult = new ApiResult<>();
        apiResult.setCode (StatusCodeEnum.SC200.getCode ());
        apiResult.setMsg (msg);
        apiResult.setData (data);
        return apiResult;
    }

    public static <T> ApiResult<T> fail (Integer code, String msg) {
        ApiResult<T> apiResult = new ApiResult<>();
        apiResult.setCode (code);
        apiResult.setMsg (msg);
        return apiResult;
    }
}

3. 全局异常捕获处理,使用 @RestControllerAdvice 注解

@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {
    /**
     * 捕获其他异常
     */
    @ResponseStatus (HttpStatus.INTERNAL_SERVER_ERROR)
    @ExceptionHandler (Exception.class)
    public ApiResult<String> handle (Exception e) {
        log.error ("全局异常信息:{}", e.getMessage ());
        return ApiResult.fail (StatusCodeEnum.SC500.getCode (), StatusCodeEnum.SC500.getMsg () + ":" + e.getMessage ());
    }
}
注解功能
@RestControllerAdviceRestController 的增强类,可用于实现全局异常处理器
@ExceptionHandler统一处理某一类异常,从而减少代码重复率和复杂度,比如要获取自定义异常可以 @ExceptionHandler (BusinessException.class)
@ResponseStatus指定客户端收到的 http 状态码

注:请求进来 会按照 filter -> interceptor -> controllerAdvice -> aspect -> controller 的顺序调用,
404 异常(NoHandlerFoundException)是无法通过这种方式捕获的,因为在 Filter 层发生的异常都会到 Spring 默认的异常处理。如果你在配置文件配置了 server.error.path 的话,就会使用你配置的异常处理地址,如果没有就会使用你配置的 error.path 路径地址,如果还是没有,默认使用 /error 来作为发生异常的处理地址。如果想要替换默认的非 Controller 异常处理直接实现 Spring 提供的 ErrorController 接口就行了。

4. 拦截 Controller 方法的返回值,统一处理返回值 / 响应体

因为我们后面每写一个接口都需要调用 ApiResult.success () 这行代码对结果进行包装,重复劳动,浪费体力,** 我们只需要实现 SpringBoot 提供的 ResponseBodyAdvice 接口即可解决这一问题。

@RestControllerAdvice
public class ApiResultWrapper implements ResponseBodyAdvice<Object> {

    /**
     * 是否支持 advice 功能
     */
    @Override
    public boolean supports (MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
        return true;
    }

    /**
     * 对返回的数据进行处理
     */
    @Override
    public Object beforeBodyWrite (Object o, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        if (o instanceof String) {
            return JSON.toJSONString (ApiResult.success (o));
        }
        // 这个判断的作用:防止全局异常处理后返回的结果(类型为 ApiResult)再次被包装
        if (o instanceof ApiResult) {
            return o;
        }
        return ApiResult.success (o);
    }
    
}

5. 创建 Controller,定义两个方法,让第二个方法抛异常

实现 ResponseBodyAdvice 接口后,方法的返回值类型就可以不再是先前统一的 ApiResult 了。

@RestController
public class TestController {

    @GetMapping ("/test1")
    public String test1 () {
        return "当前时间:" + LocalDateTime.now ().format (DateTimeFormatter.ofPattern ("yyyy-MM-dd HH:mm:ss"));
    }

    @GetMapping ("/test2")
    public Integer test2 () {
        System.out.println (1 / 0);
        return Integer.MAX_VALUE;
    }
    
}

6. 测试

分别请求 http://localhost:8080/test1、http://localhost:8080/test2,结果如下
在这里插入图片描述在这里插入图片描述
在全局异常处理类中写了一行代码

log.error ("全局异常信息:{}", e.getMessage ());

所以调用 test2 方法控制台打印异常信息如下

在这里插入图片描述

PS:欢迎转载但请注明出处:https://blog.csdn.net/weixin_43553153/article/details/119685809

  • 25
    点赞
  • 99
    收藏
    觉得还不错? 一键收藏
  • 6
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值