【Spring AOP】统⼀数据返回格式

为什么需要统一数据返回格式?

  1. ⽅便前端程序员更好的接收和解析后端数据接⼝返回的数据。
  2. 降低前端程序员和后端程序员的沟通成本,按照某个格式实现就⾏了,因为所有接⼝都是这样返回的。
  3. 有利于项⽬统⼀数据的维护和修改。
  4. 有利于后端技术部⻔的统⼀规范的标准制定,不会出现稀奇古怪的返回内容。

统一数据返回格式的实现

  1. 类上添加 @ControllerAdvice 注解表示对 Controller 的通知
  2. 实现 ResponseBodyAdvice 接口并重写 supports 和 beforeBodyWrite 方法, 拦截返回的数据, 并对数据格式进行重写
javascript 复制代码
@ControllerAdvice
public class ResponseAdvice implements ResponseBodyAdvice {

	@Resource
    private ObjectMapper objectMapper;

    /**
     * 内容是否需要重写(通过此⽅法可以选择性部分控制器和⽅法进⾏重写)
     * 返回 true 表示重写, 进入下面的 beforeBodyWrite 方法
     */
    @Override
    public boolean supports(MethodParameter returnType, Class converterType) {
        return true;
    }
    
    /**
     * ⽅法返回之前调⽤此⽅法
     */
    @SneakyThrows // 这个注解, 等价于 方法上加 Throws Exception
    @Override
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType,
                                  Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
        // 构造统⼀返回对象
        HashMap<String, Object> result = new HashMap<>();
        result.put("state", 1);
        result.put("msg", "");
        result.put("data", body);
        if (body instanceof String) {
            // 如果数据类型是 String 的话, 比较特殊, 需要 使用 jackson单独处理
            return objectMapper.writeValueAsString(request);
        }
        return result;
    }
}

统一数据返回格式举例

假如说约定返回格式为一个类 AjaxResult

javascript 复制代码
/**
 *  统一数据返回格式
 *  实现 Serializable 接口支持序列化
 */
@Data
public class AjaxResult implements Serializable {
    // 状态码
    private Integer code;
    // 状态码描述信息
    private String msg;
    // 返回的数据
    private Object data;

    /**
     *  操作成功返回的结果
     */
    public static AjaxResult success(Object data) {
        AjaxResult result = new AjaxResult();
        result.setCode(200);
        result.setMsg("");
        result.setData(data);
        return result;
    }

    /**
     * 根据需要调用不同的重载的 方法
     */
    public static AjaxResult success(Integer code, Object data) {
        AjaxResult result = new AjaxResult();
        result.setCode(code);
        result.setMsg("");
        result.setData(data);
        return result;
    }


    public static AjaxResult success(Integer code, String msg, Object data) {
        AjaxResult result = new AjaxResult();
        result.setCode(code);
        result.setMsg(msg);
        result.setData(data);
        return result;
    }

    /**
     *  操作失败返回的结果
     */
    public static AjaxResult fail(Integer code, String msg) {
        AjaxResult result = new AjaxResult();
        result.setCode(code);
        result.setMsg(msg);
        result.setData(null);
        return result;
    }


    public static AjaxResult fail(Integer code, String msg, Object data) {
        AjaxResult result = new AjaxResult();
        result.setCode(code);
        result.setMsg(msg);
        result.setData(data);
        return result;
    }
}

拦截返回的数据, 对不满足条件的数据进行封装使满足格式.

javascript 复制代码
/**
 * 实现统一数据返回的保底类
 * 说明: 在返回数据之前, 检测数据的类型是否是统一的对象, 如果不是, 封装成统一的对象
 */

@ControllerAdvice
public class ResponseAdvice implements ResponseBodyAdvice {

    @Autowired
    private ObjectMapper objectMapper;


    /**
     * 开关, 如果是 true, 才会调用 beforeBodyWrite 方法
     */
    @Override
    public boolean supports(MethodParameter returnType, Class converterType) {
        return true;
    }

    /**
     * 对数据格式进行校验和封装
     */
    @SneakyThrows  // 这个注解, 等价于 方法上加 Throws Exception
    @Override
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
        if (body instanceof AjaxResult) {
        	// 本身格式就是 AjaxResult 了
            return body;
        }
        if (body instanceof String) {
            // 是字符串的话比较特殊, 需要使用 jackson
            return objectMapper.writeValueAsString(AjaxResult.success(body));
        }
        // 如果不是字符串, 直接返回对象即可
        return AjaxResult.success(body);
    }
}

好啦! 以上就是对 统⼀数据返回格式 的讲解,希望能帮到你 !
评论区欢迎指正 !

相关推荐
欧洵.几秒前
Java.基于UDP协议的核心内容
java·开发语言·udp
xunyan62347 分钟前
第九章 JAVA常用类
java·开发语言
Smoothzjc19 分钟前
👉 求你了,别再裸写 fetch 做 AI 流式响应了!90% 的人都在踩这个坑
前端·人工智能·后端
China_Yanhy23 分钟前
AWS S3 深度配置指南:每一栏每个选项有什么作用
java·数据库·aws
秃了也弱了。32 分钟前
FASTJSON库:阿里出品java界json解析库,使用与踩坑记录
java·开发语言·json
安全渗透Hacker1 小时前
参数未校验导致的DOS(服务拒绝)问题典型场景
java·安全·web安全·网络安全·安全性测试
superman超哥1 小时前
Rust 或模式(Or Patterns)的语法:多重匹配的优雅表达
开发语言·后端·rust·编程语言·rust或模式·or patterns·多重匹配
Chan161 小时前
微服务 - Higress网关
java·spring boot·微服务·云原生·面试·架构·intellij-idea
二哈喇子!1 小时前
JavaSE 与 JavaEE 知识点整合
java·servlet·tomcat
之歆1 小时前
Spring AI入门到实战到原理源码-多模型协作智能客服系统
java·人工智能·spring