详解SpringBoot如何实现统一后端返回格式

作者:北根娃 时间:2022-11-27 05:26:24 

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

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

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

1.1 返回String

@GetMapping("/hello")
public String hello() {
   return  "hello";
}

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

hello

1.2 返回自定义对象

@GetMapping("/student")
public Student getStudent() {
       Student student = new Student();
       student.setId(1);
       student.setName("didiplus");
       return  student;
}

//student的类
@Data
public class Student {
   private Integer id;
   private String name;
}

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

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

1.3 接口异常

@GetMapping("/error")
public int error(){
   int i = 9/0;
   return i;
}

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

SpringBoot的版本是v2.6.7,

2.定义返回对象

package com.didiplus.common.web.response;

import lombok.Data;

import java.io.Serializable;

/**
* Author: didiplus
* Email: 972479352@qq.com
* CreateTime: 2022/4/24
* Desc: Ajax 返 回 JSON 结 果 封 装 数 据
*/

@Data
public class Result<T> implements Serializable {

/**
    * 是否返回成功
    */
   private boolean success;

/**
    * 错误状态
    */
   private int code;

/***
    * 错误信息
    */
   private String msg;

/**
    * 返回数据
    */
   private T data;

/**
    * 时间戳
    */
   private long timestamp ;

public Result (){
       this.timestamp = System.currentTimeMillis();
   }
   /**
    * 成功的操作
    */
   public static <T> Result<T> success() {
       return  success(null);
   }

/**
    * 成 功 操 作 , 携 带 数 据
    */
   public static <T> Result<T> success(T data){
       return success(ResultCode.RC100.getMessage(),data);
   }

/**
    * 成 功 操 作, 携 带 消 息
    */
   public static <T> Result<T> success(String message) {
       return success(message, null);
   }

/**
        * 成 功 操 作, 携 带 消 息 和 携 带 数 据
        */
   public static <T> Result<T> success(String message, T data) {
       return success(ResultCode.RC100.getCode(), message, data);
   }

/**
    * 成 功 操 作, 携 带 自 定 义 状 态 码 和 消 息
    */
   public static <T> Result<T> success(int code, String message) {
       return success(code, message, null);
   }

public static <T> Result<T> success(int code,String message,T data) {
       Result<T> result = new Result<T>();
       result.setCode(code);
       result.setMsg(message);
       result.setSuccess(true);
       result.setData(data);
       return result;
   }

/**
    * 失 败 操 作, 默 认 数 据
    */
   public static <T> Result<T> failure() {
       return failure(ResultCode.RC100.getMessage());
   }

/**
    * 失 败 操 作, 携 带 自 定 义 消 息
    */
   public static <T> Result<T> failure(String message) {
       return failure(message, null);
   }

/**
    * 失 败 操 作, 携 带 自 定 义 消 息 和 数 据
    */
   public static <T> Result<T> failure(String message, T data) {
       return failure(ResultCode.RC999.getCode(), message, data);
   }

/**
    * 失 败 操 作, 携 带 自 定 义 状 态 码 和 自 定 义 消 息
    */
   public static <T> Result<T> failure(int code, String message) {
       return failure(ResultCode.RC999.getCode(), message, null);
   }

/**
    * 失 败 操 作, 携 带 自 定 义 状 态 码 , 消 息 和 数 据
    */
   public static <T> Result<T> failure(int code, String message, T data) {
       Result<T> result = new Result<T>();
       result.setCode(code);
       result.setMsg(message);
       result.setSuccess(false);
       result.setData(data);
       return result;
   }

/**
    * Boolean 返 回 操 作, 携 带 默 认 返 回 值
    */
   public static <T> Result<T> decide(boolean b) {
       return decide(b, ResultCode.RC100.getMessage(), ResultCode.RC999.getMessage());
   }

/**
    * Boolean 返 回 操 作, 携 带 自 定 义 消 息
    */
   public static <T> Result<T> decide(boolean b, String success, String failure) {
       if (b) {
           return success(success);
       } else {
           return failure(failure);
       }
   }
}

3.定义状态码

package com.didiplus.common.web.response;

import lombok.Getter;

/**
* Author: didiplus
* Email: 972479352@qq.com
* CreateTime: 2022/4/24
* Desc: 统 一 返 回 状 态 码
*/
public enum ResultCode {
   /**操作成功**/
   RC100(100,"操作成功"),
   /**操作失败**/
   RC999(999,"操作失败"),
   /**服务限流**/
   RC200(200,"服务开启限流保护,请稍后再试!"),
   /**服务降级**/
   RC201(201,"服务开启降级保护,请稍后再试!"),
   /**热点参数限流**/
   RC202(202,"热点参数限流,请稍后再试!"),
   /**系统规则不满足**/
   RC203(203,"系统规则不满足要求,请稍后再试!"),
   /**授权规则不通过**/
   RC204(204,"授权规则不通过,请稍后再试!"),
   /**access_denied**/
   RC403(403,"无访问权限,请联系管理员授予权限"),
   /**access_denied**/
   RC401(401,"匿名用户访问无权限资源时的异常"),
   /**服务异常**/
   RC500(500,"系统异常,请稍后重试"),

INVALID_TOKEN(2001,"访问令牌不合法"),
   ACCESS_DENIED(2003,"没有权限访问该资源"),
   CLIENT_AUTHENTICATION_FAILED(1001,"客户端认证失败"),
   USERNAME_OR_PASSWORD_ERROR(1002,"用户名或密码错误"),
   UNSUPPORTED_GRANT_TYPE(1003, "不支持的认证模式");

/**自定义状态码**/
   @Getter
   private final int code;

/**
    * 携 带 消 息
    */
   @Getter
   private final String message;
   /**
    * 构 造 方 法
    */
   ResultCode(int code, String message) {

this.code = code;

this.message = message;
   }
}

4.统一返回格式

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

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

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

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

5.高级实现方式

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

5.1 ResponseBodyAdvice的源码

public interface ResponseBodyAdvice<T> {
/**
* 是否支持advice功能
* true 支持,false 不支持
*/
   boolean supports(MethodParameter var1, Class<? extends HttpMessageConverter<?>> var2);

/**
* 对返回的数据进行处理
*/
   @Nullable
   T beforeBodyWrite(@Nullable T var1, MethodParameter var2, MediaType var3, Class<? extends HttpMessageConverter<?>> var4, ServerHttpRequest var5, ServerHttpResponse var6);
}

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

@RestControllerAdvice
public class ResponseAdvice  implements ResponseBodyAdvice<Object> {

@Autowired
   ObjectMapper objectMapper;

@Override
   public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
       return true;
   }

@SneakyThrows
   @Override
   public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response)  {
       if (body instanceof  String){
           return objectMapper.writeValueAsString(Result.success(ResultCode.RC100.getMessage(),body));
       }
       return Result.success(ResultCode.RC100.getMessage(),body);
   }
}

需要注意两个地方:

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

  • 全局异常处理

  • 全局数据绑定

  • 全局数据预处理

5.2 String类型判断

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

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

@GetMapping("/hello")
   public String hello() {
       return "hello,didiplus";
   }

@GetMapping("/student")
   public Student getStudent() {
       Student student = new Student();
       student.setId(1);
       student.setName("didiplus");
       return student;
   }

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

{ "success": true, "code": 100, "msg": "操作成功", "data": "hello,didiplus", "timestamp": 1650786993454 }

来源:https://www.cnblogs.com/alanlin/p/16191008.html

标签:SpringBoot,后端,返回格式
0
投稿

猜你喜欢

  • Java数据结构之单链表详解

    2023-11-04 17:02:20
  • 详解Spring Boot 定制HTTP消息转换器

    2023-11-24 20:20:51
  • Thymeleaf 3.0 自定义标签方言属性的实例讲解

    2023-03-24 20:40:23
  • Java并发编程示例(七):守护线程的创建和运行

    2023-11-25 11:39:32
  • MyBatis自定义映射关系和关联查询实现方法详解

    2021-12-18 21:25:28
  • Java多线程yield心得分享

    2023-11-29 05:58:15
  • 老生常谈java中cookie的使用

    2023-11-11 04:37:59
  • 分别在Groovy和Java中创建并初始化映射的不同分析

    2023-11-26 08:15:42
  • 用代码更新你的jar包

    2023-09-19 23:01:04
  • 如何利用Jackson序列化忽略指定类型的属性详解

    2023-11-16 09:01:27
  • SpringBoot自动配置原理详解

    2023-08-19 09:25:55
  • 不看后悔!揭秘游戏服务器开发

    2023-08-22 16:20:30
  • C语言实现扫雷游戏(含注释详解)

    2023-11-02 15:25:07
  • 解析Java设计模式编程中命令模式的使用

    2023-11-12 04:49:45
  • Maven打包jar包没有主属性问题解决方案

    2023-11-25 06:56:32
  • 设计模式在Spring框架中的应用汇总

    2023-10-22 19:20:09
  • Java实现分页的前台页面和后台代码

    2021-07-22 17:10:04
  • 详解如何实现SpringBoot的底层注解

    2023-11-23 06:21:04
  • JDK1.8中的ConcurrentHashMap源码分析

    2023-11-27 06:02:32
  • 详解spring cloud ouath2中的资源服务器

    2022-09-24 15:36:43
  • asp之家 软件编程 m.aspxhome.com