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

网友投稿 303 2022-10-09

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

目录1.为什么要对SpringBoot返回统一的标准格式1.1 返回String1.2 返回自定义对象1.3 接口异常2.定义返回对象3.定义状态码4.统一返回格式5.高级实现方式5.1 ResponseBodyAdvice的源码5.2 String类型判断

在前后端分离的项目中后端返回的格式一定要友好,不然会对前端的开发人员带来很多的工作量。那么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 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 Result success() {

return success(null);

}

/**

* 成 功 操 作 , 携 带 数 据

*/

public static Result success(T data){

return success(ResultCode.RC100.getMessage(),data);

}

Uikti /**

* 成 功 操 作, 携 带 消 息

*/

public static Result success(String message) {

return success(message, null);

}

/**

* 成 功 操 作, 携 带 消 息 和 携 带 数 据

*/

public static Result success(String message, T data) {

return success(ResultCode.RC100.getCode(), message, data);

}

/**

* 成 功 操 作, 携 带 自 定 义 状 态 码 和 消 息

*/

public static Result success(int code, String message) {

return success(code, message, null);

}

public static Result success(int code,String message,T data) {

Result result = new Result();

result.setCode(code);

result.setMsg(message);

result.setSuccess(true);

result.setData(data);

return result;

}

/**

* 失 败 操 作, 默 认 数 据

*/

public static Result failure() {

return failure(ResultCode.RC100.getMessage());

}

/**

* 失 败 操 作, 携 带 自 定 义 消 息

*/

public static Result failure(String message) {

return failure(message, null);

}

/**

* 失 败 操 作, 携 带 自 定 义 消 息 和 数 据

*/

public static Result failure(String message, T data) {

return failure(ResultCode.RC999.getCode(), message, data);

}

/**

* 失 败 操 作, 携 带 自 定 义 状 态 码 和 自 定 义 消 息

*/

public static Result failure(int code, String message) {

return failure(ResultCode.RC999.getCode(), message, null);

}

/**

* 失 败 操 作, 携 带 自 定 义 状 态 码 , 消 息 和 数 据

*/

public static Result&http://lt;T> failure(int code, String message, T data) {

Result result = new Result();

result.setCode(code);

result.setMsg(message);

result.setSuccess(false);

result.setData(data);

return result;

}

/**

* Boolean 返 回 操 作, 携 带 默 认 返 回 值

*/

public static Result decide(boolean b) {

return decide(b, ResultCode.RC100.getMessage(), ResultCode.RC999.getMessage());

}

/**

* Boolean 返 回 操 作, 携 带 自 定 义 消 息

*/

public static Result 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("/Uiktihello")

public Result 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 {

/**

* 是否支持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 {

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

以上就是详解SpringBoot如何实现统一后端返回格式的详细内容,更多关于SpringBoot统一后端返回格式的资料请关注我们其它相关文章!

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Linux常用命令精讲(二)
下一篇:基于消息队列 RocketMQ 的大型分布式应用上云最佳实践
相关文章

 发表评论

暂时没有评论,来抢沙发吧~