使用RequestBodyAdvice、ResponseBodyAdvice优化程序请求与响应

前言

在前后端分离的项目中,前端与后台一般会约定好固定的参数和响应的数据结构,如:

参数:

public class ApiRequest<T> {

    private String token;
    
    private String version;

    /**
     * 业务参数
     */
    private T data;

}

复制代码

响应:

public class ApiResult<T> {

    /**
     * 编码,当code=0时代表成功,其他则为失败"
     */
    private Integer code;

    /**
     * 描述信息
     */
    private String msg;

    /**
     * 成功后返回的数据
     */
    private T data;

    ...

}
复制代码

一般传统的做法是在 Controller层方法直接接收ApiRequest参数和直接返回ApiResult的实例:

  • 在参数中传入ApiRequest对象,然后手动获取业务参数data进行处理

  • 每个接口手动生成ApiResult对象并返回。

这一部分工作其实是重复也无太多意义的,那么有没有一种方法可以自动做到 我们只关注 ApiRequest.data和 ApiResult.data,让程序自动将参数传入到 业务参数data中和 控制层方法只返回data,程序自动封装成ApiResult并返回呢?那么今天的主人公 RequestBodyAdvice,ResponseBodyAdvice就登场了。

RequestBodyAdvice

RequestBodyAdviceSpringMVC4.2提供的一个接口,它允许请求体被读取并转换为对象,并将处理结果对象作为@RequestBody参数或者 @HttpEntity方法参数。由此可见,它的作用范围为:

  • 使用@RequestBody进行标记的参数
  • 参数为HttpEntity

提供的方法

boolean supports(MethodParameter methodParameter, Type targetType,
      Class<? extends HttpMessageConverter<?>> converterType);
复制代码

该方法返回true时,才会进去下面的系列方法

HttpInputMessage beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter,
      Type targetType, Class<? extends HttpMessageConverter<?>> converterType) throws IOException;
复制代码

body数据读取之前调用,一般在此方法中对body数据进行修改

Object afterBodyRead(Object body, HttpInputMessage inputMessage, MethodParameter parameter,
      Type targetType, Class<? extends HttpMessageConverter<?>> converterType);
复制代码

body读取之后操作,一般直接返回原实例

Object handleEmptyBody(@Nullable Object body, HttpInputMessage inputMessage, MethodParameter parameter,
      Type targetType, Class<? extends HttpMessageConverter<?>> converterType);
复制代码

当body问empty时操作

实现步骤

  • 编写一个实现类实现RequestBodyAdvice接口
  • 分别实现对应的方法
  • 实现类上添加注解标记:ControllerAdvice

实例

@ControllerAdvice
public class RequestInterceptor implements RequestBodyAdvice {

    @Override
    public boolean supports(MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) {
        RequestAdvice requestAdvice = methodParameter.getMethodAnnotation(RequestAdvice.class);
        if (requestAdvice == null) {
            requestAdvice = methodParameter.getDeclaringClass().getAnnotation(RequestAdvice.class);
        }
        return requestAdvice != null;
    }

    @Override
    public HttpInputMessage beforeBodyRead(HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) throws IOException {
        String bodyStr = IOUtils.toString(httpInputMessage.getBody(), StandardCharsets.UTF_8);
        return new HttpInputMessage() {
            @Override
            public InputStream getBody() throws IOException {
                ApiRequest<Object> request = JsonUtils.json2Obj(bodyStr, new TypeReference<ApiRequest<Object>>() {
                });
                String body = bodyStr;
                if (request != null && request.getData() != null) {
                    body = JsonUtils.obj2Json(request.getData());
                }
                return new ByteArrayInputStream(body.getBytes(StandardCharsets.UTF_8));
            }

            @Override
            public HttpHeaders getHeaders() {
                return httpInputMessage.getHeaders();
            }
        };
    }

    @Override
    public Object afterBodyRead(Object body, HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) {
        return body;
    }

    @Override
    public Object handleEmptyBody(Object body, HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) {
        return body;
    }
复制代码

上面实例中,supports添加了支持条件:当控制层方法或者类上有标记注解 @RequestAdvice注解时,才会进入其他相关方法。当不需要任何限制时,supports直接返回true即可。

ResponseBodyAdvice

ResponseBodyAdviceSpringMVC4.1提供的一个接口,它允许在 执行 @ResponseBody后自定义返回数据,或者将返回@ResponseEntity的 Controller Method在写入主体前使用 HttpMessageConverter进行自定义操作。由此可见,它的作用范围为:

  • 使用@ResponseBody注解进行标记
  • 返回@ResponseEntity

提供的方法

  • boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType);
    复制代码

    该方法返回true时,才会进去下面的 beforeBodyWrite方法。该方法可以添加一些判断条件,比如 方法上有 xxx 注解的才会生效等等。

  • T beforeBodyWrite(@Nullable T body, MethodParameter returnType, MediaType selectedContentType,
          Class<? extends HttpMessageConverter<?>> selectedConverterType,
          ServerHttpRequest request, ServerHttpResponse response);
    复制代码

    body写入前的操作。

实现步骤

  • 编写一个实现类实现ResponseBodyAdvice接口
  • 重写supportsboforeBodyWrite
  • 实现类上添加注解标记:ControllerAdvice

实例

@ControllerAdvice
public class ResponseInterceptor implements ResponseBodyAdvice {

    @Override
    public boolean supports(MethodParameter returnType, Class converterType) {
        ResponseAdvice responseAdvice = returnType.getMethodAnnotation(ResponseAdvice.class);
        if (responseAdvice == null) {
            responseAdvice = returnType.getDeclaringClass().getAnnotation(ResponseAdvice.class);
        }
        return responseAdvice != null;
    }

    @Override
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
        try {
            ApiResult<Object> result = new ApiResult<>();
            result.setCode(0);
            result.setMsg("success");
            result.setdData(body);
            return result;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return body;
    }
}
复制代码

上面实例中,supports添加了支持条件:当控制层方法或者类上有标记注解 @ResponseAdvice直接时,才会进入beforeBodyWrite方法。当不需要任何限制时,supports直接返回true即可。

其他

上述只是针对 公共参数公共返回这一种情况对 RequestBodyAdviceResponseBodyAdvice进行了说明,当然这两种接口不止这一种应用场景,比如对参数或者返回进行加解密都可以使用这两种接口进行实现。具体使用场景用户可根据实际情况进行使用。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Spring框架中,如果我们想要对一个GET请求请求体进行处理,则可以通过实现`RequestBodyAdvice`接口来实现。 `RequestBodyAdvice`接口有四个方法: 1. `supports(MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType)`:用于判断支持哪些请求体的处理。如果返回`true`,则会调用后面三个方法对请求体进行处理;否则不会进行处理。 2. `beforeBodyRead(HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType)`:在请求体读取之前调用,用于对请求体进行处理。 3. `afterBodyRead(Object body, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType)`:在请求体读取之后调用,用于对请求体进行处理。 4. `handleError(HttpMessageNotReadableException ex, HttpInputMessage inputMessage, MethodParameter parameter, Type targetType, Class<? extends HttpMessageConverter<?>> converterType)`:在读取请求体出现异常时调用。 需要注意的是,`RequestBodyAdvice`只对请求体为JSON格式的GET请求有效,对于其他格式的请求体(如form-data),需要使用其他方式进行处理。 以下是一个使用`RequestBodyAdvice`处理请求体的示例: ```java @ControllerAdvice public class CustomRequestBodyAdvice implements RequestBodyAdvice { @Override public boolean supports(MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) { return methodParameter.getMethod().getName().equals("getUserInfo") && type.equals(UserInfo.class); } @Override public Object beforeBodyRead(HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) throws IOException, HttpMessageNotReadableException { String requestBody = StreamUtils.copyToString(httpInputMessage.getBody(), StandardCharsets.UTF_8); JSONObject jsonObject = JSON.parseObject(requestBody); UserInfo userInfo = new UserInfo(); userInfo.setUserName(jsonObject.getString("userName")); userInfo.setPassword(jsonObject.getString("password")); return userInfo; } @Override public Object afterBodyRead(Object o, HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) { return o; } @Override public Object handleEmptyBody(Object o, HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) { return o; } @Override public void handleError(HttpMessageNotReadableException httpMessageNotReadableException, HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) throws IOException { throw httpMessageNotReadableException; } } ``` 在以上示例中,我们使用`supports`方法判断请求体是否为`UserInfo`类型,如果是则调用`beforeBodyRead`方法对请求体进行处理,将JSON格式的请求体转换成`UserInfo`对象。最后,我们返回处理后的`UserInfo`对象即可。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值