首先需要明确如下几个概念:
1:web开发的时候,过滤器属于java原生组件,而拦截器属于spring框架的组件,从它们的参数就可以看出来,过滤器参数为ServletRequest, 而拦截器为HttpServeletRequest,因为spring本来就是web开发针对的就是http协议,而java则是针对所有网络通信不单单是http协议。
2:需要了解一下ServletRequest HttpServletRequest 之间的联系和区别
3:tomcat处理http请求的。Tomcat将请求转换成了RequestFacade传给过滤器,而RequestFacade实现了HttpServlestRequest接口。至于如何转换可以看(http://blog.csdn.net/aesop_wubo/article/details/7630440)
4:大部分IO输入流是无法重复读取的,只能读取一次。再读取时,会抛出IO异常。无论是get请求还是post请求,在后端读取一次之后,便无法再次读取为了解决这个问题,我们需要包装HttpServletRequest对象,缓存body里面的数据,再次读取的时候从缓存里面读取。
5: 我们可以定义自己的HttpServletRequest实现类来达到缓存的目的。
public class MAPIHttpServletRequestWrapper extends HttpServletRequestWrapper {
private static final Logger logger = LoggerFactory.getLogger(MAPIHttpServletRequestWrapper.class);
private Map<String, String[]> parameterMap; // get方法
private byte[] requestBody = null; // post 方法body数据
public MAPIHttpServletRequestWrapper(HttpServletRequest request) {
super(request);
//缓存请求body
try {
parameterMap = request.getParameterMap();
requestBody = StreamUtils.copyToByteArray(request.getInputStream());
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 获取所有参数名, get相关方法重写
*
* @return 返回所有参数名
*/
@Override
public Enumeration<String> getParameterNames() {
Vector<String> vector = new Vector<>(parameterMap.keySet());
return vector.elements();
}
/**
* 获取指定参数名的值,如果有重复的参数名,则返回第一个的值 接收一般变量 ,如text类型
*
* @param name
* 指定参数名
* @return 指定参数名的值
*/
@Override
public String getParameter(String name) {
String[] results = parameterMap.get(name);
if (results == null || results.length <= 0)
return null;
else {
System.out.println("modify before:" + results[0]);
return modify(results[0]);
}
}
/**
* 获取指定参数名的所有值的数组,如:checkbox的所有数据
* 接收数组变量 ,如checkobx类型
*/
@Override
public String[] getParameterValues(String name) {
String[] results = parameterMap.get(name);
if (results == null || results.length <= 0)
return null;
else {
int length = results.length;
for (int i = 0; i < length; i++) {
results[i] = modify(results[i]);
logger.info("modify before,{}:{}",name, results[i]);
}
return results;
}
}
/**
* 自定义的一个简单修改原参数的方法,即:给原来的参数值前面添加了一个修改标志的字符串
*
* @param string
* 原参数值
* @return 修改之后的值 ,这里并不进行改变
*/
private String modify(String string) {
return string;
}
/**
* 重写 getReader()
*/
@Override
public BufferedReader getReader() throws IOException {
return new BufferedReader(new InputStreamReader(getInputStream()));
}
/**
* 重写 getInputStream()
*/
@Override
public ServletInputStream getInputStream() throws IOException {
logger.info("modify before post");
if(requestBody == null){
requestBody= new byte[0];
}
final ByteArrayInputStream bais = new ByteArrayInputStream(requestBody);
MyServletInputStream myServletInputStream = new MyServletInputStream(bais);
return myServletInputStream;
}
// 定义自己的ServletInputStream
class MyServletInputStream extends ServletInputStream{
private InputStream inputStream;
public MyServletInputStream(InputStream inputStream){
super();
this.inputStream = inputStream;
}
@Override
public boolean isFinished() {
return false;
}
@Override
public boolean isReady() {
return false;
}
@Override
public void setReadListener(ReadListener readListener) {
return;
}
@Override
public int read() throws IOException {
return inputStream.read();
}
}
}
6:修改过滤器 dofilter方法
if(request instanceof HttpServletRequest && response instanceof HttpServletResponse){
request = new MAPIHttpServletRequestWrapper((HttpServletRequest) request);
response = new MAPIHttpServletResponseWrapper((HttpServletResponse)response);
}
将tomcat传过来的httprequest实现类RequestFacade转换成我们自己写的类即可,这样就可以实现多次读取request里面的参数值。
总结:其实除了自己可以实现之外也可以使用spring官方提供的组件ContentCachingRequestWrapper,其原理和上面讲的是一样的,只是具体实现细节上有少许不同。
参考资料:
http://tomcat.apache.org/tomcat-7.0-doc/servletapi/javax/servlet/http/HttpServletRequestWrapper.html
http://mabusyao.iteye.com/blog/1048087
https://www.zhihu.com/question/39872707
https://www.jianshu.com/p/dee45d97a4fa

本文探讨了在web开发中,如何实现多次读取HttpServletRequest的参数值。由于大部分IO输入流只能读取一次,因此需要通过包装HttpServletRequest对象并缓存数据来实现。介绍了自定义HttpServletRequest实现类以及使用Spring的ContentCachingRequestWrapper组件来达到目的。

1206

被折叠的 条评论
为什么被折叠?



