getReader()/getInputStream() has already been called for this request
项目中需要在filter对request中body中的数据进行处理,发现了这个问题
getRead() has already been called for this request/getInputStream() has already been called for this request
根本原因是
request中的getRead() 和 getInputStream()在读取一次后标记为-1,无法再次被读取
而可以看到@RequestBody在ServletServerHttpRequest中,也调用了getInputStream()方法
所以如果数据在filter中被读取,将无法在@RequestBody中再次读取。
此时解决方案,将request进行包装(装饰者模式),并重写getInputStream()和getRead()方法
public class ParamHttpServletRequestWrapper extends HttpServletRequestWrapper {
private final String body;
public ParamHttpServletRequestWrapper(HttpServletRequest request) throws IOException, ServletException {
super(request);
body = read(request);
}
@Override
public BufferedReader getReader() {
return new BufferedReader(new InputStreamReader(this.getInputStream()));
}
@Override
public ServletInputStream getInputStream() {
final ByteArrayInputStream bais = new ByteArrayInputStream(body.getBytes());
return new ServletInputStream() {
@Override
public boolean isFinished() {
return false;
}
@Override
public boolean isReady() {
return false;
}
@Override
public void setReadListener(ReadListener readListener) {
}
@Override
public int read() {
return bais.read();
}
};
}
private String read(HttpServletRequest request) throws IOException {
BufferedReader bufferedReader = request.getReader();
StringWriter writer = new StringWriter();
write(bufferedReader, writer);
return writer.getBuffer().toString();
}
private void write(Reader reader, Writer writer) throws IOException {
int read;
char[] buf = new char[1024 * 8];
while ((read = reader.read(buf)) != -1) {
writer.write(buf, 0, read);
}
}
}