Http Servlet 請求在讀取一次後會遺失POST 正文中的參數
在這種情況下,在處理POST 請求時了解這一點至關重要,參數通常在請求正文中發送。預設情況下,Servlet 容器在第一次存取參數時會消耗輸入字元流或二進位流。因此,後續嘗試在過濾器鏈中存取它們會失敗。
要解決這個問題,需要考慮兩種主要方法:
1。攔截器組件:
作為使用過濾器鏈的替代方案,您可以使用攔截器組件。這些元件可以使用方面來實現,並允許對解析的請求正文進行操作,而無需依賴所使用的請求輸入流。這種方法通常更有效,因為請求 InputStream 僅解析一次以建立您自己的模型物件。
2. HttpServletRequestWrapper:
對於 HTTP 層首選過濾器鏈的情況,擴展 HttpServletRequestWrapper 是一個有效的方法。此技術涉及消耗請求 InputStream 並且快取位元組:
public class MultiReadHttpServletRequest extends HttpServletRequestWrapper { private ByteArrayOutputStream cachedBytes; public MultiReadHttpServletRequest(HttpServletRequest request) { super(request); } @Override public ServletInputStream getInputStream() throws IOException { if (cachedBytes == null) cacheInputStream(); return new CachedServletInputStream(cachedBytes.toByteArray()); } @Override public BufferedReader getReader() throws IOException{ return new BufferedReader(new InputStreamReader(getInputStream())); } private void cacheInputStream() throws IOException { /* Cache the inputstream in order to read it multiple times. For * convenience, I use apache.commons IOUtils */ cachedBytes = new ByteArrayOutputStream(); IOUtils.copy(super.getInputStream(), cachedBytes); } /* An input stream which reads the cached request body */ private static class CachedServletInputStream extends ServletInputStream { private final ByteArrayInputStream buffer; public CachedServletInputStream(byte[] contents) { this.buffer = new ByteArrayInputStream(contents); } @Override public int read() { return buffer.read(); } @Override public boolean isFinished() { return buffer.available() == 0; } @Override public boolean isReady() { return true; } @Override public void setReadListener(ReadListener listener) { throw new RuntimeException("Not implemented"); } } }
透過在將原始請求傳遞到過濾器鏈之前使用 MultiReadHttpServletRequest 包裝原始請求,可以多次讀取請求正文。請注意,此解決方案還有助於透過 getParameterXXX 方法進行多次讀取。
編輯:
對於較新版本的 ServletInputStream 接口,需要使用 isReady 和 setReadListener 等其他方法實施的。請參閱此問題以獲取更多資訊。
以上是為什麼我的 Http Servlet 請求在一次讀取後會遺失 POST 正文參數?的詳細內容。更多資訊請關注PHP中文網其他相關文章!