Springboot 之重复读取 Request Body 内容

简介

在项目中,有两个场景会用到从 RequestBody 中读取内容。

  • 打印请求日志

  • 提供 Api 接口,在 Api 方法执行之前,从 RequestBody 中读取参数进行验签,验签通过后再执行 Api 方法的具体逻辑

这时就需要对 RequestBoy 进行重复读取。要实现对 RequestBody 进行重复读取的思路如下:

  • 继承 HttpServletRequestWrapper 包装类,读取 RequestBody 的内容,然后缓存到 byte[] 数组 中;

  • 增加过滤器,将包装类加入过滤器链中;

自定义 HttpServletRequestWrapper 包装类

由于采用采用application/json传输参数时,HttpServletRequest只能读取一次 body 中的内容。因为是读的字节流,读完就没了,因此需要需要做特殊处理。为实现述多次读取 Request 中的 Body 内容,需继承 HttpServletRequestWrapper 类,读取 Body 的内容,然后缓存到 byte[] 中;这样就可以实现多次读取 Body 的内容了。

package com.olive.request; 
import org.apache.commons.io.IOUtils; 
import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.*;
  
public class RequestWrapper extends HttpServletRequestWrapper {
    
    //参数字节数组
    private byte[] requestBody;
    
    //Http请求对象
    private HttpServletRequest request; 
    
    public RequestWrapper(HttpServletRequest request) throws IOException {
        super(request);
        this.request = request;
    }
  
    @Override
    public ServletInputStream getInputStream() throws IOException {
        /**
         * 每次调用此方法时将数据流中的数据读取出来,然后再回填到InputStream之中
         * 解决通过@RequestBody和@RequestParam(POST方式)读取一次后控制器拿不到参数问题
         */
        if (null == this.requestBody) {
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            IOUtils.copy(request.getInputStream(), baos);
            this.requestBody = baos.toByteArray();
        }
  
        final ByteArrayInputStream bais = new ByteArrayInputStream(requestBody);
        return new ServletInputStream() {
  
            @Override
            public boolean isFinished() {
                return false;
            }
  
            @Override
            public boolean isReady() {
                return false;
            }
  
            @Override
            public void setReadListener(ReadListener listener) {
  
            }
  
            @Override
            public int read() {
                return bais.read();
            }
        };
    }
  
    public byte[] getRequestBody() {
        return requestBody;
    }
  
    @Override
    public BufferedReader getReader() throws IOException {
        return new BufferedReader(new InputStreamReader(this.getInputStream()));
    }
}

增加过滤器,转换Request为自定义Request

package com.olive.filter;  

import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;

public class RequestBodyReaderFilter implements Filter {
   
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    
    }
  
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) {
        try {
            ServletRequest requestWrapper = null;
            if (request instanceof HttpServletRequest) {
                requestWrapper = new RequestWrapper((HttpServletRequest) request);
            }
            if (requestWrapper == null) {
                chain.doFilter(request, response);
            } else {
                chain.doFilter(requestWrapper, response);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } catch (ServletException e) {
            e.printStackTrace();
        } 
    }
  
    @Override
    public void destroy() {
    
    } 
}

注册Filter

package com.olive.config;

import java.util.ArrayList;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import com.olive.filter.RequestBodyReaderFilter

@Configuration
public class WebConfig {
  
   @Bean
   public FilterRegistrationBean<RequestBodyReaderFilter> fequestBodyReaderFilter() {
      FilterRegistrationBean<RequestBodyReaderFilter> registrationBean = new FilterRegistrationBean<>();
      RequestBodyReaderFilter filter = new RequestBodyReaderFilter();
      registrationBean.setFilter(filter);
      ArrayList<String> urls = new ArrayList<>();
      urls.add("/*");//配置过滤规则
      registrationBean.setUrlPatterns(urls);
      registrationBean.setOrder(3);
      return registrationBean;
   }
   
}

记得点「」和「在看」↓

爱你们

  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot中,重写请求体可以通过自定义RequestBodyAdvice和ResponseBodyAdvice来实现。 首先,创建一个类实现RequestBodyAdvice接口,实现beforeBodyRead方法,该方法将在读取请求体之前被调用。在此方法中,您可以对请求体进行修改或转换,并返回更改后的请求体。以下是一个示例: ```java @ControllerAdvice public class CustomRequestBodyAdvice implements RequestBodyAdvice { @Override public boolean supports(MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) { return true; } @Override public Object handleEmptyBody(Object o, HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) { return o; } @Override public HttpInputMessage beforeBodyRead(HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) throws IOException { String requestBody = IOUtils.toString(httpInputMessage.getBody(), StandardCharsets.UTF_8); // 修改或转换请求体 String modifiedRequestBody = "modified " + requestBody; return new CustomHttpInputMessage(new ByteArrayInputStream(modifiedRequestBody.getBytes()), httpInputMessage.getHeaders()); } @Override public Object afterBodyRead(Object o, HttpInputMessage httpInputMessage, MethodParameter methodParameter, Type type, Class<? extends HttpMessageConverter<?>> aClass) { return o; } } ``` 然后,创建一个类实现ResponseBodyAdvice接口,实现beforeBodyWrite方法,该方法将在写入响应体之前被调用。在此方法中,您可以对响应体进行修改或转换,并返回更改后的响应体。以下是一个示例: ```java @ControllerAdvice public class CustomResponseBodyAdvice implements ResponseBodyAdvice<Object> { @Override public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) { return true; } @Override public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) { if (o instanceof String) { // 修改或转换响应体 String modifiedResponseBody = "modified " + o; return modifiedResponseBody; } return o; } } ``` 注意:以上示例中的CustomHttpInputMessage是自定义类,用于封装修改后的请求体。这个类的实现可以参考以下示例: ```java public class CustomHttpInputMessage implements HttpInputMessage { private InputStream body; private HttpHeaders headers; public CustomHttpInputMessage(InputStream body, HttpHeaders headers) { this.body = body; this.headers = headers; } @Override public InputStream getBody() throws IOException { return body; } @Override public HttpHeaders getHeaders() { return headers; } } ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值