如何解決HttpServletRequest.getInputStream()多次讀取問題
HttpServletRequest.getInputStream()多次讀取問題
使用POST方法發(fā)送數(shù)據(jù)時,我們習慣于把數(shù)據(jù)包裝成json格式。

有些情況下,我們會在Filter中讀取body數(shù)據(jù)進行數(shù)據(jù)校驗,GET方法獲取參數(shù)比較簡單。
對于POST方法,可使用如下方法從request中獲取body參數(shù):
private String getBody(HttpServletRequest request) throws IOException {
InputStream in = request.getInputStream();
BufferedReader br = new BufferedReader(new InputStreamReader(in, Charset.forName("UTF-8")));
StringBuffer sb = new StringBuffer("");
String temp;
while ((temp = br.readLine()) != null) {
sb.append(temp);
}
if (in != null) {
in.close();
}
if (br != null) {
br.close();
}
return sb.toString();
}注意,這里有了一次request.getInputStream()調(diào)用。
但是在測試時,一直報JSON格式不正確的錯誤。經(jīng)調(diào)查發(fā)現(xiàn),項目中使用了公司基礎(chǔ)組件中的Filter,而該Filter中也解析了body。
同時,不出所料,也是通過調(diào)用getInputStream()方法獲取的。
原來:
- 一個InputStream對象在被讀取完成后,將無法被再次讀取,始終返回-1;
- InputStream并沒有實現(xiàn)reset方法(可以重置首次讀取的位置),無法實現(xiàn)重置操作;
因此,當自己寫的Filter中調(diào)用了一次getInputStream()后,后面再調(diào)用getInputStream()讀取的數(shù)據(jù)都為空,所以才報JSON格式不正確的錯誤。
解決方法
- 緩存數(shù)據(jù)
- 使用
HttpServletRequestWrapper進行包裝
緩存數(shù)據(jù)
所謂緩存數(shù)據(jù),其實就是調(diào)用ServletRequest的setAttribute(String s, Object o)來存儲數(shù)據(jù)。
1.獲取到body后,直接緩存
String body = getBody(request);
request.setAttribute("body", body);優(yōu)點:
方便
缺點:
不能控制第三方Filter
1.其他地方需要使用body時,只需調(diào)用getAttribute方法就能獲取數(shù)據(jù)了:
request.getAttribute("body");HttpServletRequestWrapper包裝
public class RequestWrapper extends HttpServletRequestWrapper {
private final byte[] body;
public RequestWrapper(HttpServletRequest request) throws IOException {
super(request);
body = getBodyStringFromReq(request).getBytes(Charset.forName("UTF-8"));
}
public String getBodyString() {
try {
return new String(body, "UTF-8");
} catch (UnsupportedEncodingException ex) {
return new String(body);
}
}
private String getBodyStringFromReq(ServletRequest request) {
StringBuilder sb = new StringBuilder();
InputStream inputStream = null;
BufferedReader reader = null;
try {
inputStream = request.getInputStream();
reader = new BufferedReader(new InputStreamReader(inputStream, Charset.forName("UTF-8")));
String line;
while ((line = reader.readLine()) != null) {
sb.append(line);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (inputStream != null) {
try {
inputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (reader != null) {
try {
reader.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return sb.toString();
}
}在Filter中使用時,FilterChain.doFilter()傳入Wrapper對象:
public class TestFilter implements Filter {
@Override
public void destroy() {
}
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
RequestWrapper requestWrapper = new RequestWrapper((HttpServletRequest)request);
String body = requestWrapper.getBodyString();
chain.doFilter(requestWrapper, response); //傳入Wrapper對象
}
@Override
public void init(FilterConfig arg0) throws ServletException {
}
}這樣,位于后面的Filter就可以擁有唯一一次調(diào)用HttpServletRequest.getInputStream()的機會了。
優(yōu)點:
不影響第三方Filter
缺點:
多寫了這么多代碼,麻煩了一些
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
java應(yīng)用占用內(nèi)存過高排查的解決方案
這篇文章主要介紹了java應(yīng)用占用內(nèi)存過高排查的解決方案,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-03-03
SpringBoot利用jpa連接MySQL數(shù)據(jù)庫的方法
這篇文章主要介紹了SpringBoot利用jpa連接MySQL數(shù)據(jù)庫的方法,本文通過示例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-10-10
java 較大數(shù)據(jù)量取差集,list.removeAll性能優(yōu)化詳解
這篇文章主要介紹了java 較大數(shù)據(jù)量取差集,list.removeAll性能優(yōu)化詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09

