HttpURLConnection模擬瀏覽器http請求【轉】


本文是參考網絡文章測試整理,報歉忘記參考地址了,要是作者有意見就請聯系我.文中只是對程序的運行做了測試,但理論沒有去考證正確性,等后面研究下再完善.這只是做參考 

  1 import java.io.BufferedReader;
  2 import java.io.DataOutputStream;
  3 import java.io.InputStreamReader;
  4 import java.net.HttpURLConnection;
  5 import java.net.URL;
  6 import java.net.URLEncoder;
  7 
  8 /**
  9  * HttpURLConnection 模擬瀏覽器http請求處理測試
 10  * 
 11  * 整體操作流程(注意順序)
 12  *   1.根據url地址創建URL對象
 13  *   2.url.openConnection() 得到HttpURLConnection
 14  *   3.設置HttpURLConnection的配置(會根據配置生成http請求頭信息)
 15  *   4.httpConn.connect()建立與服務器的TCP連接
 16  *   5.通過httpConn得到輸出流,並寫入參數
 17  *   6.通過httpConn得到輸入流,並讀取服務器返回信息(這時才真正產生http請求發送給服務器,但可以根據配置來改變;一般返回可能是html字符串)
 18  * 
 19  * httpRequestPost/httpRequestGet兩種方式測試通過,返回的是html字符串(后台是jsp)
 20  *
 21  */
 22 public class HttpURLConnectionTest {
 23 
 24     public static void main(String[] strs) throws Exception{
 25         HttpURLConnectionTest t = new HttpURLConnectionTest();
 26         String url = "http://localhost:8080/URLTest/filesystem/fileBrowse.do";
 27         t.httpUrlConnection(url);
 28     }
 29     
 30     private void httpUrlConnection(String urlStr) throws Exception{
 31         httpRequestGet(urlStr);
 32         httpRequestPost(urlStr);
 33     }
 34     
 35     /**
 36      * http post請求
 37      * @param urlStr URL地址
 38      * @throws Exception
 39      */
 40     private void httpRequestPost(String urlStr) throws Exception{
 41         URL url = new URL(urlStr);
 42         
 43         HttpURLConnection httpConn = (HttpURLConnection)url.openConnection();
 44         
 45         //設置是否向connection輸出,因為是post請求,參數要放在 http正文內,因此需要設為true
 46         httpConn.setDoOutput(true);
 47         
 48         //是否向connection輸入,默認為true
 49         httpConn.setDoInput(true);
 50         
 51         //POST請求方式
 52         httpConn.setRequestMethod("POST");
 53         
 54         //post請求不能使用緩存
 55         httpConn.setUseCaches(false);
 56         
 57         //是否自動執行重定向,默認為true
 58         httpConn.setInstanceFollowRedirects(true);
 59         
 60         //配置本次連接的Content-type,配置為application/x-www-form-urlencoded的意思是正文是urlencoded編碼過的form參數
 61         //下面我們可以看到我們對正文內容使用URLEncoder.encode 進行編碼
 62         httpConn.setRequestProperty("Content-Type", "application/x-www-form-urlencoded");
 63         
 64         //連接,從openConnection()至此的配置必須要在connect之前完成, 注意的是connection.getOutputStream會隱含的進行connect。
 65         httpConn.connect();
 66         
 67         DataOutputStream dos = new DataOutputStream(httpConn.getOutputStream());
 68         
 69         //發送正文,正文內容其實跟get的URL中'?'后的參數字符串一致 word=wait&tn=news&from=news
 70         String content = "filePath=" + URLEncoder.encode("設計資料/設計", "utf-8");
 71         
 72         //DataOutputStream.writeBytes將字符串中的16位的unicode字符以8位的字符形式寫道流里面
 73         dos.writeBytes(content); 
 74         
 75         //刷新流
 76         dos.flush();
 77         
 78         //關閉流
 79         dos.close();
 80         
 81         // 取得輸入流,並使用Reader讀取
 82         BufferedReader reader = new BufferedReader(new InputStreamReader(httpConn.getInputStream()));
 83         
 84         System.out.println("=========post request接收數據內容開始============");
 85         String lines;
 86         while ((lines = reader.readLine()) != null) {
 87             System.out.println(lines);
 88         }
 89         reader.close();
 90         System.out.println("=========post request接收數據內容結束============");
 91         httpConn.disconnect();
 92     }
 93 
 94     /**
 95      * http get 請求
 96      * @param urlStr 請求URL地址
 97      * @throws Exception
 98      */
 99     private void httpRequestGet(String urlStr) throws Exception{
100         //URL拼接,如:"http://www.baidu.com?name=HI,中國",這里對特殊字符進行了編碼,不然會產生亂碼
101         URL url = new URL(urlStr + "?filePath=" + URLEncoder.encode("設計資料/設計", "utf-8"));
102         
103         //openConnection函數會根據URL的協議返回不同的URLConnection子類的對象
104         //這里URL是一個http,因此實際返回的是HttpURLConnection 
105         HttpURLConnection httpConn = (HttpURLConnection)url.openConnection();
106         
107         //進行連接,實際上request要在下一句的connection.getInputStream()函數中才會真正發到 服務器****待驗證
108         httpConn.connect();
109         
110          // 取得輸入流,並使用Reader讀取
111         BufferedReader reader = new BufferedReader(new InputStreamReader(httpConn.getInputStream()));
112         
113         System.out.println("=========get request接收數據內容開始============");
114         String lines;
115         while ((lines = reader.readLine()) != null) {
116             System.out.println(lines);
117         }
118         reader.close();
119         System.out.println("=========get request接收數據內容結束============");
120         httpConn.disconnect();
121     }
122 
123     
124     /**
125      * http post 請求(未測試)
126      * 
127      * @param urlStr
128      * @throws Exception
129      */
130     public static void httpRequestPostBig(String urlStr) throws Exception {
131             URL postUrl = new URL(urlStr);
132             HttpURLConnection connection = (HttpURLConnection) postUrl
133                     .openConnection();
134             connection.setDoOutput(true);
135             connection.setDoInput(true);
136             connection.setRequestMethod("POST");
137             connection.setUseCaches(false);
138             connection.setInstanceFollowRedirects(true);
139             connection.setRequestProperty("Content-Type","application/x-www-form-urlencoded");
140              //與httpRequestPost()的不同,設置了塊大小為5字節
141             connection.setChunkedStreamingMode(5);
142             connection.connect();
143             /*
144              * 注意,下面的getOutputStream函數工作方式於在httpRequestPost()里面的不同
145              * 在httpRequestPost()里面該函數仍在准備http request,沒有向服務器發送任何數據
146              * 而在這里由於設置了ChunkedStreamingMode,getOutputStream函數會根據connect之前的配置
147              * 生成http request頭,先發送到服務器。
148              */
149             DataOutputStream out = new DataOutputStream(connection
150                     .getOutputStream());
151             String content = "name=" + URLEncoder.encode("發送到服務器                                                                             " +
152                     "                                          " +
153                     "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa", "utf-8");
154             out.writeBytes(content); 
155 
156             out.flush();
157             out.close(); //到此時服務器已經收到了完整的http request了,而在httpRequestPost()函數里,要等到下一句getInputStream()服務器才能收到http請求。
158             BufferedReader reader = new BufferedReader(new InputStreamReader(
159                     connection.getInputStream()));
160             
161             out.flush();
162             out.close();
163             String line;
164             System.out.println("==========post request開始==========");
165             while ((line = reader.readLine()) != null) {
166                 System.out.println(line);
167             }
168             System.out.println("==========post request結束==========");
169             reader.close();
170             connection.disconnect();
171         }
172 }

 

HttpURLConnection.connect函數,實際上只是建立了一個與服務器的tcp連接,並沒有實際發送http請求。無論是post還是get,http請求實際上直到HttpURLConnection.getInputStream()這個函數里 

面才正式發送出去。 
  在httpRequestPost() 中,順序很重要,對connection對象的一切配置(那一堆set函數)都必須要在connect()函數執行之前完成。而對 outputStream的寫操作,又必須要在inputStream的 

讀操作之前。這些順序實際上是由http請求的格式決定的。 
  http 請求實際上由兩部分組成,一個是http頭,所有關於此次http請求的配置都在http頭里面定義,一個是正文content,在connect()函數里面,會根據HttpURLConnection對象的配置值生 

成http頭,因此在調用connect函數之前,就必須把所有的配置准備好。 
  緊接着http頭的是http請求的正文,正文的內容通過outputStream寫入,實際上outputStream不是一個網絡流,充其量是個字符串流,往里面寫入的東西不會立即發送到網絡,而是在流關閉 

后,根據輸入的內容生成http正文。 
  至此,http請求的東西已經准備就緒。在getInputStream()函數調用的時候,就會把准備好的http請求正式發送到服務器了,然后返回一個輸入流,用於讀取服務器對於此次http請求的返回 

信息。由於http請求在getInputStream的時候已經發送出去了(包括http頭和正文),因此在getInputStream()函數之后對connection對象進行設置(對http頭的信息進行修改)或者寫入 outputStream 

(對正文進行修改)都是沒有意義的了,執行這些操作會導致異常的發生 
  post請求的OutputStream實際上不是網絡流,而是寫入內存,在getInputStream中才真正把寫道流里面的內容作為正文與根據之前的配置生成的http request頭合並成真正的http request, 

並在此時才真正向服務器發送。 
  HttpURLConnection.setChunkedStreamingMode 函數可以改變這個模式,設置了ChunkedStreamingMode后,不再等待OutputStream關閉后生成完整的http request一次過發送,而是先發送 

http request頭,正文內容則是網路流的方式實時傳送到服務器。實際上是不告訴服務器http正文的長度,這種模式適用於向服務器傳送較大的或者是不容易獲取長度的數據,如文件。

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM