spring-boot中统一修改http响应体内容的方法

2021-09-25  本文已影响0人  向上生长之路

在spring-boot开发中,有时候会遇到一些统一修改http接口响应体内容的场景,比如统一增加一些响应头信息、统一增加一些数据包装处理、数据转换等场景,spring框架中已经为我们提供了这种能力,即ResponseBodyAdvice切面的使用,ResponseBodyAdvice就是spring框架中预留的钩子,它作用在Controller方法执行完成之后,http响应体写回客户端之前,这个时候我们就能方便的织入一些自己的业务逻辑处理了

工作原理

查看ResponseBodyAdvice接口的源码中,提供了两个方法

public interface ResponseBodyAdvice<T> {
    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);
}

使用方式

  1. 自定义一个实现ResponseBodyAdvice接口的实现类,添加注解@ControllerAdvice,来拦截Controller控制器的执行

示例代码:https://github.com/netbuffer/spring-boot-demo/blob/master/src/main/java/cn/netbuffer/springboot/demo/controller/CustomResponseBodyAdvice.java

@Slf4j
@ControllerAdvice
public class CustomResponseBodyAdvice implements ResponseBodyAdvice {

    @Override
    public boolean supports(MethodParameter methodParameter, Class aClass) {
        //根据需要来判断是否允许执行beforeBodyWrite
        Class targetClass = methodParameter.getMethod().getDeclaringClass();
        log.debug("supports execute methodParameter={} targetClass={} class={}", methodParameter, targetClass, aClass);
        return true;
    }

    @Override
    public Object beforeBodyWrite(Object data, MethodParameter methodParameter, MediaType mediaType, Class aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        //data即为Controller方法执行后写回的值
        log.debug("beforeBodyWrite data={}", data);
        //这里可以拿到request信息来做特定的逻辑处理
        if (serverHttpRequest.getURI().getPath().equals("/ua")) {
            //测试添加http响应头
            serverHttpResponse.getHeaders().set("x-abc", "header-value");
        }
        //测试根据不同数据类型,来做不同的包装处理、转换等操作
        if (data instanceof String) {
            return "[" + data + "]";
        } else if (data instanceof Map) {
            ((Map) data).put("extData", "value");
            return data;
        } else {
            return data;
        }
    }

}
  1. 建立测试用的控制器UserAgentController,用于返回User-Agent头信息
    示例代码:https://github.com/netbuffer/spring-boot-demo/blob/master/src/main/java/cn/netbuffer/springboot/demo/controller/UserAgentController.java
@RestController
@RequestMapping("/ua")
public class UserAgentController {

    @GetMapping
    public String ua(@RequestHeader("User-Agent") String userAgent) {
        return userAgent;
    }
}
  1. 建立测试用于返回Map类型数据的映射方法
    完整示例代码:https://github.com/netbuffer/spring-boot-demo/blob/master/src/main/java/cn/netbuffer/springboot/demo/controller/BeanController.java
    @GetMapping(value = {"map"})
    public Map getMap() {
        Map data = new HashMap();
        data.put("name", "test");
        data.put("age", "30");
        return data;
    }

运行效果

使用Postman工具访问/ua接口测试
可以看到返回的数据中已经被[]包装了


再查看返回的http响应头中,能看到我们添加的x-abc响应头

再访问map接口查看结果,可以看到有多余的extData字段被添加进去了

完整测试工程参考:https://github.com/netbuffer/spring-boot-demo

上一篇下一篇

猜你喜欢

热点阅读