Spring MVC 实现request请求体多次读取

项目中有些场景可能需要多次读取requestBody的内容,如日志打印,参数校验,而HttpServletRequest对象中的数据流只能读取一次,再次读取就会抛出异常,因此我们需要自定义request包装类实现多次读取请求体内容。

正常情况下,controller接口只能申明一次@RequestBody,写多个会报错。原因就是request请求体的流对象只能单次读取。

1.自定义包装类

public class RequestWrapper extends HttpServletRequestWrapper {

    //参数字节数组
    private byte[] requestBody;

    //Http请求对象
    private HttpServletRequest request;

    public RequestWrapper(HttpServletRequest request) {
        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()));
    }
}

2. 新增过滤器对HttpServletRequest进行包装

@WebFilter(urlPatterns = "/*")
public class RequestWrapperFilter implements Filter {
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        try {
            if(!(servletRequest instanceof HttpServletRequest)){
                filterChain.doFilter(servletRequest, servletResponse);
                return;
            }
            //将改为我们定义的包装类
            ServletRequest requestWrapper = new RequestWrapper((HttpServletRequest) servletRequest);
            filterChain.doFilter(requestWrapper, servletResponse);
        } catch (IOException | ServletException e) {
            e.printStackTrace();
        }
    }
}

3. 测试

经过以上配置,接口参数就可以写多个RequestBody了

@PostMapping("/post")
public Object postTest(@RequestBody Book book,@RequestBody Book book2){
    log.info("controller----->,{}",book);
    return book;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring MVC中的@RequestBody注解可以将HTTP请求中的参数映射到Java对象中,其中包括Map类型。通常,我们可以使用对象模型来获取请求中的参数。例如,我们可以使用@RequestBody注解将请求中的JSON数据映射为一个Java对象,如下所示: ``` @RequestMapping(value = "/api/somepath", method = RequestMethod.POST) public ResponseEntity<?> someMethod(@RequestBody SomeClass someClass) { // do something with someClass object … } ``` 但是,如果请求头中传递的数据较为复杂,我们可能需要将数据映射为Map类型,这样我们就可以直接读取并使用其中的字段。例如,假设我们有以下的JSON数据: ``` { "name": "John", "age": 30, "address": { "street": "123 Main St", "city": "Anytown" } } ``` 我们可以将该请求映射到一个Map类型中: ``` @RequestMapping(value = "/api/somepath", method = RequestMethod.POST) public ResponseEntity<?> someMethod(@RequestBody Map<String, Object> requestBody) { // access request body fields with map String name = (String) requestBody.get("name"); Integer age = (Integer) requestBody.get("age"); Map<String, String> address = (Map<String, String>) requestBody.get("address"); String street = address.get("street"); String city = address.get("city"); … } ``` 这样,我们可以通过Map类型的键对直接获取请求中的参数,从而简化代码的编写和阅读。但需要注意的是,当Map中的需要进行类型转换时,我们需要使用强制转换。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值