如何解決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()
調用。
但是在測試時,一直報JSON格式不正確的錯誤。經調查發(fā)現(xiàn),項目中使用了公司基礎組件中的Filter,而該Filter中也解析了body。
同時,不出所料,也是通過調用getInputStream()
方法獲取的。
原來:
- 一個InputStream對象在被讀取完成后,將無法被再次讀取,始終返回-1;
- InputStream并沒有實現(xiàn)reset方法(可以重置首次讀取的位置),無法實現(xiàn)重置操作;
因此,當自己寫的Filter中調用了一次getInputStream()
后,后面再調用getInputStream()
讀取的數(shù)據(jù)都為空,所以才報JSON格式不正確的錯誤。
解決方法
- 緩存數(shù)據(jù)
- 使用
HttpServletRequestWrapper
進行包裝
緩存數(shù)據(jù)
所謂緩存數(shù)據(jù),其實就是調用ServletRequest
的setAttribute(String s, Object o)
來存儲數(shù)據(jù)。
1.獲取到body后,直接緩存
String body = getBody(request); request.setAttribute("body", body);
優(yōu)點:
方便
缺點:
不能控制第三方Filter
1.其他地方需要使用body時,只需調用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就可以擁有唯一一次調用HttpServletRequest.getInputStream()
的機會了。
優(yōu)點:
不影響第三方Filter
缺點:
多寫了這么多代碼,麻煩了一些
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
SpringBoot利用jpa連接MySQL數(shù)據(jù)庫的方法
這篇文章主要介紹了SpringBoot利用jpa連接MySQL數(shù)據(jù)庫的方法,本文通過示例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-10-10java 較大數(shù)據(jù)量取差集,list.removeAll性能優(yōu)化詳解
這篇文章主要介紹了java 較大數(shù)據(jù)量取差集,list.removeAll性能優(yōu)化詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09