大家經常 用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>
request.getInputStream() 大多數用於文件上傳等