多次读取请求request里数据

本文介绍了一种在WebService中解决重复读取POST请求数据的问题方法。通过使用HttpServletRequestWrapper类重写相关方法,使得filter和controller都能正确读取content-type为“text/xml”的数据,避免了数据流只能被读取一次的限制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题:
webService通信时,在filter中拦截请求做用户权限验证。拦截到content-type的内容是“text/xml”,只能用传入的数据流中取数据了,而request数据流里的数据只能读取一次,filter里读取了,到了controller里就会读取失败。
解决方法:

  • 采用ThreadLocal,在filter里把读取到的post参数存入ThreadLocal里,而controller就可以再从ThreadLocal里把请求参数读取出来
  • 使用servlet提供的HttpServletRequestWrapper类,重写相关ServletRequest方法,实现多次读取的能力。

这里只记录重写HttpServletRequestWrapper的方法。

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.*;
 
/**
 * Created by Endstart on 16/11/30.
 */
public class WrappedHttpServletRequest extends HttpServletRequestWrapper {
 
    private byte[] bytes;
    private WrappedServletInputStream wrappedServletInputStream;
 
    public WrappedHttpServletRequest(HttpServletRequest request) throws IOException {
        super(request);
        // 读取输入流里的请求参数,并保存到bytes里
        bytes = IOUtils.toByteArray(request.getInputStream());
        ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(bytes);
        this.wrappedServletInputStream = new WrappedServletInputStream(byteArrayInputStream);

     // 很重要,把post参数重新写入请求流
     reWriteInputStream();
 
    }
 
    /**
     * 把参数重新写进请求里
     */
    public void reWriteInputStream() {
        wrappedServletInputStream.setStream(new ByteArrayInputStream(bytes != null ? bytes : new byte[0]));
    }
 
    @Override
    public ServletInputStream getInputStream() throws IOException {
        return wrappedServletInputStream;
    }
 
    @Override
    public BufferedReader getReader() throws IOException {
        return new BufferedReader(new InputStreamReader(wrappedServletInputStream));
    }
 
    /**
     * 获取post参数,可以自己再转为相应格式
     */
    public String getRequestParams() throws IOException {
        return new String(bytes, this.getCharacterEncoding());
    }
 
    private class WrappedServletInputStream extends ServletInputStream {
 
        public void setStream(InputStream stream) {
            this.stream = stream;
        }
 
        private InputStream stream;
 
        public WrappedServletInputStream(InputStream stream) {
            this.stream = stream;
        }
 
        @Override
        public int read() throws IOException {
            return stream.read();
        }
 
        @Override
        public boolean isFinished() {
            return true;
        }
 
        @Override
        public boolean isReady() {
            return true;
        }
 
        @Override
        public void setReadListener(ReadListener readListener) {
 
        }
    }
}  

filter中使用重写的HttpServletRequestWrapper来获取数据(需要根据自己的实际情况判断哪些请求需要拦截)

WrappedHttpServletRequest requestWrapper = new WrappedHttpServletRequest((HttpServletRequest) servletRequest);
String params = requestWrapper.getRequestParams();

当然放行的时候,参数也要改写为新的requestWrapper

filterChain.doFilter(requestWrapper, servletResponse);

参考文章

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值