1. 程式人生 > >使用HttpClient模擬POST請求

使用HttpClient模擬POST請求

       HttpClient 是 Apache Jakarta Common 下的子專案,可以用來提供高效的、最新的、功能豐富的支援 HTTP 協議的客戶端程式設計工具包,並且它支援 HTTP 協議最新的版本和建議。當前官網最新版介紹頁是:http://hc.apache.org/httpcomponents-client-4.5.x/index.html

       許多需要後臺模擬請求的系統或者框架都用的是httpclient。所以作為一個java開發人員,有必要學一學。本文提供了一個簡單的demo,供初學者參考。

       使用HttpClient傳送請求、接收響應很簡單,一般需要如下幾步即可:

    1. 建立CloseableHttpClient物件。
    2. 建立請求方法的例項,並指定請求URL。如果需要傳送GET請求,建立HttpGet物件;如果需要傳送POST請求,建立HttpPost物件。
    3. 如果需要傳送請求引數,可可呼叫setEntity(HttpEntity entity)方法來設定請求引數。setParams方法已過時(4.4.1版本)。
    4. 呼叫HttpGet、HttpPost物件的setHeader(String name, String value)方法設定header資訊,或者呼叫setHeaders(Header[] headers)設定一組header資訊。
    5. 呼叫CloseableHttpClient物件的execute(HttpUriRequest request)傳送請求,該方法返回一個CloseableHttpResponse。
    6. 呼叫HttpResponse的getEntity()方法可獲取HttpEntity物件,該物件包裝了伺服器的響應內容。程式可通過該物件獲取伺服器的響應內容;呼叫CloseableHttpResponse的getAllHeaders()、getHeaders(String name)等方法可獲取伺服器的響應頭。
    7. 釋放連線。無論執行方法是否成功,都必須釋放連線

       具體程式碼如下(HttpClient-4.4.1):

  1. /**  
  2.  * 簡單httpclient例項 
  3.  *  
  4.  * @author arron 
  5.  * @date 2015年11月11日 下午6:36:49  
  6.  * @version 1.0 
     
  7.  */
  8. publicclass SimpleHttpClientDemo {  
  9.     /** 
  10.      * 模擬請求 
  11.      *  
  12.      * @param url       資源地址 
  13.      * @param map   引數列表 
  14.      * @param encoding  編碼 
  15.      * @return 
  16.      * @throws ParseException 
  17.      * @throws IOException 
  18.      */
  19.     publicstatic String send(String url, Map<String,String> map,String encoding) throws ParseException, IOException{  
  20.         String body = "";  
  21.         //建立httpclient物件
  22.         CloseableHttpClient client = HttpClients.createDefault();  
  23.         //建立post方式請求物件
  24.         HttpPost httpPost = new HttpPost(url);  
  25.         //裝填引數
  26.         List<NameValuePair> nvps = new ArrayList<NameValuePair>();  
  27.         if(map!=null){  
  28.             for (Entry<String, String> entry : map.entrySet()) {  
  29.                 nvps.add(new BasicNameValuePair(entry.getKey(), entry.getValue()));  
  30.             }  
  31.         }  
  32.         //設定引數到請求物件中
  33.         httpPost.setEntity(new UrlEncodedFormEntity(nvps, encoding));  
  34.         System.out.println("請求地址:"+url);  
  35.         System.out.println("請求引數:"+nvps.toString());  
  36.         //設定header資訊
  37.         //指定報文頭【Content-type】、【User-Agent】
  38.         httpPost.setHeader("Content-type""application/x-www-form-urlencoded");  
  39.         httpPost.setHeader("User-Agent""Mozilla/4.0 (compatible; MSIE 5.0; Windows NT; DigExt)");  
  40.         //執行請求操作,並拿到結果(同步阻塞)
  41.         CloseableHttpResponse response = client.execute(httpPost);  
  42.         //獲取結果實體
  43.         HttpEntity entity = response.getEntity();  
  44.         if (entity != null) {  
  45.             //按指定編碼轉換結果實體為String型別
  46.             body = EntityUtils.toString(entity, encoding);  
  47.         }  
  48.         EntityUtils.consume(entity);  
  49.         //釋放連結
  50.         response.close();  
  51.         return body;  
  52.     }  
  53. }  

       在main方法中測試一下:

  1. publicstaticvoid main(String[] args) throws ParseException, IOException {  
  2.     String url="http://php.weather.sina.com.cn/iframe/index/w_cl.php";  
  3.     Map<String, String> map = new HashMap<String, String>();  
  4.     map.put("code""js");  
  5.     map.put("day""0");  
  6.     map.put("city""上海");  
  7.     map.put("dfc""1");  
  8.     map.put("charset""utf-8");  
  9.     String body = send(url, map,"utf-8");  
  10.     System.out.println("交易響應結果:");  
  11.     System.out.println(body);  
  12.     System.out.println("-----------------------------------");  
  13.     map.put("city""北京");  
  14.     body = send(url, map, "utf-8");  
  15.     System.out.println("交易響應結果:");  
  16.     System.out.println(body);  
  17. }  
       結果如下:
  1. 請求地址:http://php.weather.sina.com.cn/iframe/index/w_cl.php  
  2. 請求引數:[dfc=1charset=utf-8, day=0code=jscity=上海]  
  3. 交易響應結果:  
  4. (function(){var w=[];w['上海']=[{s1:'小雨',s2:'小雨',f1:'xiaoyu',f2:'xiaoyu',t1:'21',t2:'16',p1:'≤3',p2:'≤3',d1:'南風',d2:'北風'}];var add={now:'2015-11-16 13:16:23',time:'1447650983',update:'北京時間11月16日08:10更新',error:'0',total:'1'};window.SWther={w:w,add:add};})();//0  
  5. -----------------------------------  
  6. 請求地址:http://php.weather.sina.com.cn/iframe/index/w_cl.php  
  7. 請求引數:[dfc=1charset=utf-8, day=0code=jscity=北京]  
  8. 交易響應結果:  
  9. (function(){var w=[];w['北京']=[{s1:'多雲',s2:'多雲',f1:'duoyun',f2:'duoyun',t1:'9',t2:'1',p1:'≤3',p2:'≤3',d1:'無持續風向',d2:'無持續風向'}];var add={now:'2015-11-16 13:18:35',time:'1447651115',update:'北京時間11月16日08:10更新',error:'0',total:'1'};window.SWther={w:w,add:add};})();//0  

       現在我們測試一下https連結:https://www.qingyidai.com/investmanagement/invest.shtml

  1. publicstaticvoid main(String[] args) throws ParseException, IOException {  
  2.     String url = "https://www.qingyidai.com/investmanagement/invest.shtml";  
  3.     String body = send(url, null"utf-8");  
  4.     System.out.println("交易響應結果:");  
  5.     System.out.println(body);  
  6. }  
       結果發現,居然正常拿到結果了:


       原來如果網站的證書已經被ca機構認證通過了,那麼用HttpClient來呼叫的話,會直接成功的。不用再單獨配置htts連結了。不過如果是自簽名的證書,還是需要配置https的