本文是參考網絡文章測試整理,報歉忘記參考地址了,要是作者有意見就請聯系我.文中只是對程序的運行做了測試,但理論沒有去考證正確性,等后面研究下再完善.這只是做參考
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正文的長度,這種模式適用於向服務器傳送較大的或者是不容易獲取長度的數據,如文件。