SpringBoot中如何实现后端统一返回数据格式的最佳实践?

2026-05-06 02:322阅读0评论SEO教程
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计1926个文字,预计阅读时间需要8分钟。

SpringBoot中如何实现后端统一返回数据格式的最佳实践?

在前后端分离的项目中,后端返回的格式一定要友好,否则会对前端开发人员的工作带来很大负担。那么,SpringBoot如何实现统一的后端返回格式呢?今天我们来探讨一下。

SpringBoot可以通过自定义响应体来统一后端返回格式。具体步骤如下:

1. 创建一个响应体类,包含状态码、消息和返回数据等属性。

javapublic class Response { private int code; private String message; private Object data;

// 省略构造方法、getters和setters}

2. 创建一个响应体工具类,用于生成统一的响应体。

javapublic class ResponseUtil { public static Response success(Object data) { Response response=new Response(); response.setCode(200); response.setMessage(操作成功); response.setData(data); return response; }

public static Response error(int code, String message) { Response response=new Response(); response.setCode(code); response.setMessage(message); response.setData(null); return response; }}

3. 在控制器中使用响应体工具类返回数据。

java@RestControllerpublic class UserController { @GetMapping(/user/{id}) public Response getUserById(@PathVariable int id) { User user=userService.getUserById(id); if (user !=null) { return ResponseUtil.success(user); } else { return ResponseUtil.error(404, 用户不存在); } }}

通过以上步骤,SpringBoot可以实现统一的后端返回格式,提高前后端开发效率。

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

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

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

返回String

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

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

hello

返回自定义对象

@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"}

接口异常

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

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

SpringBoot的版本是v2.6.7,

定义返回对象

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); } } } 定义状态码

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; } } 统一返回格式

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

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

SpringBoot中如何实现后端统一返回数据格式的最佳实践?

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

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

高级实现方式

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

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注解的增强,可以实现三个方面的功能:

  1. 全局异常处理
  2. 全局数据绑定
  3. 全局数据预处理
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 }

本文共计1926个文字,预计阅读时间需要8分钟。

SpringBoot中如何实现后端统一返回数据格式的最佳实践?

在前后端分离的项目中,后端返回的格式一定要友好,否则会对前端开发人员的工作带来很大负担。那么,SpringBoot如何实现统一的后端返回格式呢?今天我们来探讨一下。

SpringBoot可以通过自定义响应体来统一后端返回格式。具体步骤如下:

1. 创建一个响应体类,包含状态码、消息和返回数据等属性。

javapublic class Response { private int code; private String message; private Object data;

// 省略构造方法、getters和setters}

2. 创建一个响应体工具类,用于生成统一的响应体。

javapublic class ResponseUtil { public static Response success(Object data) { Response response=new Response(); response.setCode(200); response.setMessage(操作成功); response.setData(data); return response; }

public static Response error(int code, String message) { Response response=new Response(); response.setCode(code); response.setMessage(message); response.setData(null); return response; }}

3. 在控制器中使用响应体工具类返回数据。

java@RestControllerpublic class UserController { @GetMapping(/user/{id}) public Response getUserById(@PathVariable int id) { User user=userService.getUserById(id); if (user !=null) { return ResponseUtil.success(user); } else { return ResponseUtil.error(404, 用户不存在); } }}

通过以上步骤,SpringBoot可以实现统一的后端返回格式,提高前后端开发效率。

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

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

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

返回String

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

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

hello

返回自定义对象

@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"}

接口异常

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

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

SpringBoot的版本是v2.6.7,

定义返回对象

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); } } } 定义状态码

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; } } 统一返回格式

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

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

SpringBoot中如何实现后端统一返回数据格式的最佳实践?

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

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

高级实现方式

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

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注解的增强,可以实现三个方面的功能:

  1. 全局异常处理
  2. 全局数据绑定
  3. 全局数据预处理
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 }