大家經常 用servlet和jsp,但是對 request.getInputStream()和request.getReader()比較陌生。
request.getParameter()
request.getInputStream()
request.getReader()
這 三個方法都是從request對象中得到提交的數據,但是用途不同,要根據<form>表單提交數據的編碼方式選擇不同的方法。
HTML中的form表單有一個關鍵屬性 enctype=application/x-www-form-urlencoded 或multipart/form-data。
enctype=application/x- www-form-urlencoded是默認的編碼方式,這種編碼方式很簡單,編碼后的結果通常是field1=value2&field2=value2&… 的形式,如 name=aaaa&Submit=Submit。這種編碼的具體規則可以在 rfc2231 里查到, 通常使用的表單也 是采用這種方式編碼的,Servlet 的 API 提供了對這種 編碼方式解碼的支持,只需要調用 ServletRequest 類中的getParameter()方法就可 以得到用戶表單中的字段和數據。
這種編碼方式( application/x-www-form-urlencoded )雖然簡單,但對於傳輸大塊的二進制數據顯得力不從心,對於傳輸這類數據,瀏覽器 采用了另一種編碼方式,即 "multipart/form-data" 的編碼方式,采用這種方式,瀏覽器可以很容易將表單內的數據和文件放在一起發送。這 種編碼方式先定義好一個不可能在數據中出現的字符串作為 分界符,然后用它將各個數據段分開,而對於每個數據段都對應着 HTML 頁面表單 中的一個 Input 區,包括一個 content-disposition 屬性,說明了這個數據段的一些信息,如果這個數據段的內容是一個文件,還會有 Content-Type 屬性,然后就是數據本身,如果以這種方式提交數據就要用request.getInputStream()或request.getReader()得到 提交的數據 ,用 request.getParameter()是得不到提交的數據的。
通過下面的代碼可以輸出采用 multipart/form-data的編碼提交的數據內容:
//1 int len = request.getContentLength(); byte buffer[] = new byte[len]; //2 InputStream in = request.getInputStream(); int total = 0; int once = 0; while ((total < len) && (once >=0)) { once = in.read(buffer,total,len); total += once; } //3 OutputStream out=new BufferedOutputStream(new FileOutputStream("c:\\Receive.log",true)); byte[] breaker="\r\nNewLog: -------------------->\r\n".getBytes(); System.out.println(request.getContentType()); out.write(breaker,0,breaker.length); out.write(buffer); out.close();
從 指定的文件( Receive.log )中可以看到如下的內容:
-----------------------------7d137a26e18
Content-Disposition: form-data; name="name"
123
-----------------------------7d137a26e18
Content-Disposition: form-data; name="introduce"
I am...
I am..
-----------------------------7d137a26e18
Content-Disposition: form-data; name="file3"; filename="C:\Autoexec.bat"
Content-Type: application/octet-stream
@echo off
prompt $d $t [ $p ]$_$$
SET PATH=d:\pf\IBMVJava2\eab\bin;%PATH%;D:\PF\ROSE98I\COMMON
-----------------------------7d137a26e18--
上面是用 IE 進行測試的結果,通過request.getInputStream()或request.getReader()可以得到form表單中提交的數據,但 是還要對數據進行 分 析才能得到form表單提交的每個參數的值。
最后注意 request.getParameter()、 request.getInputStream()、request.getReader()這三種方法是有沖突的,因為流只能被讀一次。
比如:
當form表單內容采用 enctype=application/x-www-form-urlencoded編碼時,先通過調用request.getParameter() 方法得到參數后,再調用 request.getInputStream()或request.getReader()已經得不到流中的內容,因為在調用 request.getParameter()時系統可能對表單中提交的數 據以流的形式讀了一次,反之亦然。
當form表單內容采用 enctype=multipart/form-data編碼時,即使先調用request.getParameter()也得不到數據,但是這時調用 request.getParameter()方法對 request.getInputStream()或request.getReader()沒有沖突,即使已經調用了 request.getParameter()方法也 可以通過調用request.getInputStream()或request.getReader()得 到表單中的數據,而request.getInputStream()和request.getReader()在同 一個響應中是不能混合使用的,如果混合使用就會拋異常。
附帶servlet源碼:
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("----------post--------------"); int len = request.getContentLength(); byte buffer[] = new byte[len]; InputStream in = request.getInputStream(); int total = 0; int once = 0; while ((total < len) && (once >=0)) { once = in.read(buffer,total,len); total += once; } OutputStream out=new BufferedOutputStream(new FileOutputStream("c:\\Receive.log",true)); byte[] breaker="\r\nNewLog: -------------------->\r\n".getBytes(); System.out.println(request.getContentType()); out.write(breaker,0,breaker.length); out.write(buffer); out.close(); in.close(); }
jsp源碼:
<form action="http://localhost:8080/server/servlet/ReceiveServlet" method="post" enctype="multipart/form-data"> <input name="aa"/> <input type="file" name="myfile"> <input value="提交" type="submit"/> </form>