HttpRequest不能重复读取问题学习笔记

问题背景
  1. 编写Filter再controller获取数据之前读取数据
    @Order(1)
    @Component
    public class HelloFilter extends OncePerRequestFilter {
        @Override
        protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
            byte [] bytes = new byte[1024] ;
            request.getInputStream().read(bytes);
            String content = new String(bytes, "UTF-8");
            log.info("过滤器拿到请求数据 :{}", content);
            filterChain.doFilter(request, response);
        }
    }
    
  2. 编写Controller业务方法从body中获取数据
    @RestController
    public class HomeController {
        @PostMapping("/hello")
        @ResponseBody
        public User hello(@RequestBody  User user){
            return user;
        }
        @Data
        static class User{
            private String id ;
            private String name ;
        }
    }
    
  3. 运行发现hello方法中获取到的user属性值id和name都为null
解决方式
  1. 在所有所有调用request.getInputStream()方法之前编写Filter,替换原HttpServletRequest
    @Order(0)
    @Component
    public class RequestBufferedFilter extends OncePerRequestFilter {
        @Override
        protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
            request = new BufferedHttpRequest(request) ;
            filterChain.doFilter(request, response);
        }
    }
    
  2. 编写HttpServletRequestWrapper的实现类, 重点是重写了getInputStream()方法
    public class BufferedHttpRequest extends HttpServletRequestWrapper {
        private byte[] buffer;
        private ByteArrayInputStream byteArrayInputStream;
        public BufferedHttpRequest(HttpServletRequest request) {
            super(request);
            // 将buffer填充
            try {
                log.info("从request拿出inputStream...");
                // 使用工具类将inputStream的内容拿到并转换为byte数据
                buffer = IOUtils.toByteArray(request.getInputStream());
            } catch (IOException e) {
                log.error("error : ", e);
            }
        }
        //重点: 重写getInputStream方法
        @Override
        public ServletInputStream getInputStream() throws IOException {
            log.info("将byte数组封装进BufferedServletInputStream");
            // 这里可以做我们数据的校验逻辑,比如处理XSS攻击
            log.info("getInputStream方法中处理xss攻击");
            return new BufferedServletInputStream(buffer);
        }
        
        class BufferedServletInputStream extends ServletInputStream {
            private ByteArrayInputStream byteArrayInputStream;
            public BufferedServletInputStream(byte[] buffer) {
                this.byteArrayInputStream = new ByteArrayInputStream(buffer);
            }
            public BufferedServletInputStream(ByteArrayInputStream inputStream) {
                this.byteArrayInputStream = inputStream;
            }
           //最主要重写此方法,用ByteArrayInputStream去接收
            @Override
            public int read() throws IOException {
                return byteArrayInputStream.read();
            }
            @Override
            public int read(byte[] b, int off, int len) throws IOException {
                return byteArrayInputStream.read(b, off, len);
            }
            @Override
            public synchronized void reset() throws IOException {
                byteArrayInputStream.reset();
            }
            @Override
            public boolean markSupported() {
                return true;
            }
            @Override
            public boolean isFinished() {
                return false;
            }
            @Override
            public boolean isReady() {
                return false;
            }
            @Override
            public void setReadListener(ReadListener listener) {}
        }
    }
    
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值