SpringBoot 返回统一格式

在前后端分离的项目中后端返回的格式一定要友好,不然会对前端的开发人员带来很多的工作量。那么SpringBoot如何做到统一的后端返回格式呢?今天我们一起来看看。

为什么要对SpringBoot返回统一的标准格式

在默认情况下,SpringBoot的返回格式常见的有三种:

返回String

  1. @GetMapping("/hello")
  2. public String hello() {
  3. return "hello";
  4. }

此时调用接口获取到的返回值是这样:

hello

返回自定义对象

  1. @GetMapping("/student")
  2. public Student getStudent() {
  3. Student student = new Student();
  4. student.setId(1);
  5. student.setName("didiplus");
  6. return student;
  7. }
  8. //student的类
  9. @Data
  10. public class Student {
  11. private Integer id;
  12. private String name;
  13. }

此时调用接口获取到的返回值是这样:

{"id":1,"name":"didiplus"}

接口异常

  1. @GetMapping("/error")
  2. public int error(){
  3. int i = 9/0;
  4. return i;
  5. }

此时调用接口获取到的返回值是这样:

SpringBoot的版本是v2.6.7,

定义返回对象

  1. package com.didiplus.common.web.response;
  2. import lombok.Data;
  3. import java.io.Serializable;
  4. /**
  5. * Author: didiplus
  6. * Email: 972479352@qq.com
  7. * CreateTime: 2022/4/24
  8. * Desc: Ajax 返 回 JSON 结 果 封 装 数 据
  9. */
  10. @Data
  11. public class Result<T> implements Serializable {
  12. /**
  13. * 是否返回成功
  14. */
  15. private boolean success;
  16. /**
  17. * 错误状态
  18. */
  19. private int code;
  20. /***
  21. * 错误信息
  22. */
  23. private String msg;
  24. /**
  25. * 返回数据
  26. */
  27. private T data;
  28. /**
  29. * 时间戳
  30. */
  31. private long timestamp ;
  32. public Result (){
  33. this.timestamp = System.currentTimeMillis();
  34. }
  35. /**
  36. * 成功的操作
  37. */
  38. public static <T> Result<T> success() {
  39. return success(null);
  40. }
  41. /**
  42. * 成 功 操 作 , 携 带 数 据
  43. */
  44. public static <T> Result<T> success(T data){
  45. return success(ResultCode.RC100.getMessage(),data);
  46. }
  47. /**
  48. * 成 功 操 作, 携 带 消 息
  49. */
  50. public static <T> Result<T> success(String message) {
  51. return success(message, null);
  52. }
  53. /**
  54. * 成 功 操 作, 携 带 消 息 和 携 带 数 据
  55. */
  56. public static <T> Result<T> success(String message, T data) {
  57. return success(ResultCode.RC100.getCode(), message, data);
  58. }
  59. /**
  60. * 成 功 操 作, 携 带 自 定 义 状 态 码 和 消 息
  61. */
  62. public static <T> Result<T> success(int code, String message) {
  63. return success(code, message, null);
  64. }
  65. public static <T> Result<T> success(int code,String message,T data) {
  66. Result<T> result = new Result<T>();
  67. result.setCode(code);
  68. result.setMsg(message);
  69. result.setSuccess(true);
  70. result.setData(data);
  71. return result;
  72. }
  73. /**
  74. * 失 败 操 作, 默 认 数 据
  75. */
  76. public static <T> Result<T> failure() {
  77. return failure(ResultCode.RC100.getMessage());
  78. }
  79. /**
  80. * 失 败 操 作, 携 带 自 定 义 消 息
  81. */
  82. public static <T> Result<T> failure(String message) {
  83. return failure(message, null);
  84. }
  85. /**
  86. * 失 败 操 作, 携 带 自 定 义 消 息 和 数 据
  87. */
  88. public static <T> Result<T> failure(String message, T data) {
  89. return failure(ResultCode.RC999.getCode(), message, data);
  90. }
  91. /**
  92. * 失 败 操 作, 携 带 自 定 义 状 态 码 和 自 定 义 消 息
  93. */
  94. public static <T> Result<T> failure(int code, String message) {
  95. return failure(ResultCode.RC999.getCode(), message, null);
  96. }
  97. /**
  98. * 失 败 操 作, 携 带 自 定 义 状 态 码 , 消 息 和 数 据
  99. */
  100. public static <T> Result<T> failure(int code, String message, T data) {
  101. Result<T> result = new Result<T>();
  102. result.setCode(code);
  103. result.setMsg(message);
  104. result.setSuccess(false);
  105. result.setData(data);
  106. return result;
  107. }
  108. /**
  109. * Boolean 返 回 操 作, 携 带 默 认 返 回 值
  110. */
  111. public static <T> Result<T> decide(boolean b) {
  112. return decide(b, ResultCode.RC100.getMessage(), ResultCode.RC999.getMessage());
  113. }
  114. /**
  115. * Boolean 返 回 操 作, 携 带 自 定 义 消 息
  116. */
  117. public static <T> Result<T> decide(boolean b, String success, String failure) {
  118. if (b) {
  119. return success(success);
  120. } else {
  121. return failure(failure);
  122. }
  123. }
  124. }

定义状态码

  1. package com.didiplus.common.web.response;
  2. import lombok.Getter;
  3. /**
  4. * Author: didiplus
  5. * Email: 972479352@qq.com
  6. * CreateTime: 2022/4/24
  7. * Desc: 统 一 返 回 状 态 码
  8. */
  9. public enum ResultCode {
  10. /**操作成功**/
  11. RC100(100,"操作成功"),
  12. /**操作失败**/
  13. RC999(999,"操作失败"),
  14. /**服务限流**/
  15. RC200(200,"服务开启限流保护,请稍后再试!"),
  16. /**服务降级**/
  17. RC201(201,"服务开启降级保护,请稍后再试!"),
  18. /**热点参数限流**/
  19. RC202(202,"热点参数限流,请稍后再试!"),
  20. /**系统规则不满足**/
  21. RC203(203,"系统规则不满足要求,请稍后再试!"),
  22. /**授权规则不通过**/
  23. RC204(204,"授权规则不通过,请稍后再试!"),
  24. /**access_denied**/
  25. RC403(403,"无访问权限,请联系管理员授予权限"),
  26. /**access_denied**/
  27. RC401(401,"匿名用户访问无权限资源时的异常"),
  28. /**服务异常**/
  29. RC500(500,"系统异常,请稍后重试"),
  30. INVALID_TOKEN(2001,"访问令牌不合法"),
  31. ACCESS_DENIED(2003,"没有权限访问该资源"),
  32. CLIENT_AUTHENTICATION_FAILED(1001,"客户端认证失败"),
  33. USERNAME_OR_PASSWORD_ERROR(1002,"用户名或密码错误"),
  34. UNSUPPORTED_GRANT_TYPE(1003, "不支持的认证模式");
  35. /**自定义状态码**/
  36. @Getter
  37. private final int code;
  38. /**
  39. * 携 带 消 息
  40. */
  41. @Getter
  42. private final String message;
  43. /**
  44. * 构 造 方 法
  45. */
  46. ResultCode(int code, String message) {
  47. this.code = code;
  48. this.message = message;
  49. }
  50. }

统一返回格式

  1. @GetMapping("/hello")
  2. public Result<String> hello() {
  3. return Result.success("操作成功","hello");
  4. }

此时调用接口获取到的返回值是这样:

  1. {"success":true,"code":100,"msg":"操作成功","data":"hello","timestamp":1650785058049}

这样确实已经实现了我们想要的结果,我在很多项目中看到的都是这种写法,在Controller层通过Result.success()对返回结果进行包装后返回给前端。这样显得不够专业而且不够优雅。 所以呢我们需要对代码进行优化,目标就是不要每个接口都手工制定Result返回值。

高级实现方式

要优化这段代码很简单,我们只需要借助SpringBoot提供的ResponseBodyAdvice即可。

ResponseBodyAdvice的源码:

  1. public interface ResponseBodyAdvice<T> {
  2. /**
  3. * 是否支持advice功能
  4. * true 支持,false 不支持
  5. */
  6. boolean supports(MethodParameter var1, Class<? extends HttpMessageConverter<?>> var2);
  7. /**
  8. * 对返回的数据进行处理
  9. */
  10. @Nullable
  11. T beforeBodyWrite(@Nullable T var1, MethodParameter var2, MediaType var3, Class<? extends HttpMessageConverter<?>> var4, ServerHttpRequest var5, ServerHttpResponse var6);
  12. }

只需要编写一个具体实现类即可

  1. @RestControllerAdvice
  2. public class ResponseAdvice implements ResponseBodyAdvice<Object> {
  3. @Autowired
  4. ObjectMapper objectMapper;
  5. @Override
  6. public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
  7. return true;
  8. }
  9. @SneakyThrows
  10. @Override
  11. public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
  12. if (body instanceof String){
  13. return objectMapper.writeValueAsString(Result.success(ResultCode.RC100.getMessage(),body));
  14. }
  15. return Result.success(ResultCode.RC100.getMessage(),body);
  16. }
  17. }

需要注意两个地方:

@RestControllerAdvice注解 @RestControllerAdvice是@RestController注解的增强,可以实现三个方面的功能:

  1. 全局异常处理
  2. 全局数据绑定
  3. 全局数据预处理

String类型判断

  1. if (body instanceof String){
  2. return objectMapper.writeValueAsString(Result.success(ResultCode.RC100.getMessage(),body));
  3. }

这段代码一定要加,如果Controller直接返回String的话,SpringBoot是直接返回,故我们需要手动转换成json。 经过上面的处理我们就再也不需要通过ResultData.success()来进行转换了,直接返回原始数据格式,SpringBoot自动帮我们实现包装类的封装。

  1. @GetMapping("/hello")
  2. public String hello() {
  3. return "hello,didiplus";
  4. }
  5. @GetMapping("/student")
  6. public Student getStudent() {
  7. Student student = new Student();
  8. student.setId(1);
  9. student.setName("didiplus");
  10. return student;
  11. }

此时我们调用接口返回的数据结果为:

  1. {
  2. "success": true,
  3. "code": 100,
  4. "msg": "操作成功",
  5. "data": "hello,didiplus",
  6. "timestamp": 1650786993454
  7. }
 
posted @ 2023-02-24 17:00  甜菜波波  阅读(365)  评论(0编辑  收藏  举报