本節摘要:采用httpclient插件的post方式發送二進制數據流到http服務端
說明,本節在我之前的一篇博客get/post方式調用http接口的基礎上來介紹,本節的代碼和之前的這篇博客放在同一個工程目錄的同一個包下。
1.背景介紹
最近項目中發布了一個HTTP服務,HTTP服務端采用二進制流來接收數據,
與其他系統的同事聯調的時候,總是出現一些問題,現把這些問題列出來,並在代碼中得到了解決。
(1)請求的中文,服務器端接收到后為亂碼
(2)客戶端接收到服務器端的響應報文后,中文為亂碼
(3)請求的報文中沒有設置字符集,服務器報空指針異常
(4)客戶端發送請求后,服務器端接收到的二進制數據的長度為0,報NegativeArraySizeExceptio異常
(5)服務端接收到客戶端的報文后里面全是百分號以及英文字母(不用說,你懂的。)
之前介紹過采用JDK自帶的java.net.HttpURLConnection(見博客get/post發送HTTP請求2)可以解決這個問題,但是項目中發現客戶端很多同事都是采用的httpclient插件,那咱這次就用httpclient插件來發送post請求,並解決以上問題。服務端接收代碼以及輸出代碼不能改,那么我們只能想千方設百計的去修改完善我們客戶端的代碼。
2.代碼如下:
之前的博客get/post方式調用http接口,這個直接采用發送字符串的形式,也沒有解決中文問題,
廢話咱就不多說了,直接上代碼
(1)修改web.xml配置文件,增加以下幾行

1 <!-- 模擬HTTP的調用,寫的一個只處理POST請求的http服務 --> 2 <servlet> 3 <servlet-name>PostHttpServer</servlet-name> 4 <servlet-class>httpcall.PostHttpServer</servlet-class> 5 </servlet> 6 7 <servlet-mapping> 8 <servlet-name>PostHttpServer</servlet-name> 9 <url-pattern>/postHttpServer</url-pattern> 10 </servlet-mapping>
servlet的配置,很easy
(2)HTTP服務器端的代碼

1 package httpcall; 2 3 import java.io.IOException; 4 import java.io.InputStream; 5 import javax.servlet.ServletException; 6 import javax.servlet.http.HttpServlet; 7 import javax.servlet.http.HttpServletRequest; 8 import javax.servlet.http.HttpServletResponse; 9 10 /** 11 * 12 *Module: PostHttpServer.java 13 *Description: 模擬的一個Http服務,處理客戶端的post請求 14 *Company: 15 *Author: pantp 16 *Date: May 5, 2012 17 */ 18 public class PostHttpServer extends HttpServlet { 19 20 private static final long serialVersionUID = 1L; 21 22 /* 23 * 代碼邏輯分以下三部分: 24 * 1.獲得請求報文 25 * 2.根據請求報文的信息去做業務邏輯,然后封裝返回報文 26 * 3.輸出相應報文 27 */ 28 protected void doPost(HttpServletRequest request, 29 HttpServletResponse response) throws ServletException, IOException { 30 String inJson = null;// 保存HTTP客戶端請求報文 31 String outJson = null;// 保存HTTP服務端輸出報文 32 33 // 獲得輸人報文然后打印出來 34 inJson = getInJson(request); 35 System.out 36 .println("\nauthor<pantp>===========服務端日志----POST方式接收HTTP請求,HTTP服務端收到的請求報文如下:==========\n"); 37 System.out.println(inJson); 38 System.out 39 .println("\nauthor<pantp>=================================================================\n"); 40 41 // 以下代碼部分獲得請求報文,然后去做校驗,轉換以及其他的調用其他的業務邏輯等,這里就不管它 42 // ........................................................................ 43 44 // 下面部分是輸出部分的處理 45 outJson = "{\"Response\": {\"code\": \"0\",\"message\": \"成功\",\"data\": \"12345\"}}";// 輸出不部分也以JSON格式的字符串輸出,這里我就寫死 46 47 response.setContentType("application/json; charset=UTF-8"); 48 response.getWriter().print(outJson); 49 } 50 51 // 獲得請求的報文,並作簡單的校驗 52 public String getInJson(HttpServletRequest request) throws IOException { 53 54 byte buffer[] = new byte[64 * 1024]; 55 InputStream in = request.getInputStream();// 獲取輸入流對象 56 57 int len = in.read(buffer); 58 // 必須對數組長度進行判斷,否則在new byte[len]會報NegativeArraySizeException異常 59 if (len < 0) { 60 throw new IOException("請求報文為空"); 61 } 62 63 String encode = request.getCharacterEncoding();// 獲取請求頭編碼 64 // 必須對編碼進行校驗,否則在new String(data, encode);會報空指針異常 65 if (null == encode || encode.trim().length() < 0) { 66 throw new IOException("請求報文未指明請求編碼"); 67 } 68 69 byte data[] = new byte[len]; 70 71 // 把buffer數組的值復制到data數組 72 System.arraycopy(buffer, 0, data, 0, len); 73 74 // 通過使用指定的 charset 解碼指定的 byte 數組,構造一個新的 String 75 String inJson = new String(data, encode); 76 77 return inJson; 78 } 79 80 // 不提供get的處理方式 81 protected void doGet(HttpServletRequest request, HttpServletResponse response) 82 throws ServletException, IOException { 83 } 84 }
(3)HTTP客戶端的代碼

1 package httpcall; 2 3 import java.io.IOException; 4 import org.apache.commons.httpclient.HttpClient; 5 import org.apache.commons.httpclient.HttpException; 6 import org.apache.commons.httpclient.HttpStatus; 7 import org.apache.commons.httpclient.methods.ByteArrayRequestEntity; 8 import org.apache.commons.httpclient.methods.EntityEnclosingMethod; 9 import org.apache.commons.httpclient.methods.PostMethod; 10 import org.apache.commons.httpclient.methods.RequestEntity; 11 12 /** 13 * 14 *Module: PostHttpClient.java 15 *Description: 采用httpclient插件的post方式發送流二進制流數據到HTTP服務端 16 *Company: 17 *Author: pantp 18 *Date: May 5, 2012 19 */ 20 public class PostHttpClient { 21 22 /** 23 * 發送post請求,客戶端采用二進制流發送,服務端采用二進制流介紹 24 * @param json 入參的json格式的報文 25 * @param url http服務器的地址 26 * @return 返回響應信息 27 */ 28 public static String postHttpReq(String json,String url) { 29 HttpClient httpClient = new HttpClient(); 30 31 byte b[] = json.getBytes();//把字符串轉換為二進制數據 32 RequestEntity requestEntity = new ByteArrayRequestEntity(b); 33 34 EntityEnclosingMethod postMethod = new PostMethod(); 35 postMethod.setRequestEntity(requestEntity);// 設置數據 36 postMethod.setPath(url);// 設置服務的url 37 postMethod.setRequestHeader("Content-Type", "text/html;charset=GBK");// 設置請求頭編碼 38 39 // 設置連接超時 40 httpClient.getHttpConnectionManager().getParams().setConnectionTimeout( 41 5 * 1000); 42 // 設置讀取超時 43 httpClient.getHttpConnectionManager().getParams().setSoTimeout(20 * 1000); 44 45 String responseMsg = ""; 46 int statusCode = 0; 47 try { 48 statusCode = httpClient.executeMethod(postMethod);// 發送請求 49 responseMsg = postMethod.getResponseBodyAsString();// 獲取返回值 50 } catch (HttpException e) { 51 e.printStackTrace(); 52 } catch (IOException e) { 53 e.printStackTrace(); 54 } finally { 55 postMethod.releaseConnection();// 釋放連接 56 } 57 58 if (statusCode != HttpStatus.SC_OK) { 59 System.out.println("HTTP服務異常" + statusCode); 60 } 61 return responseMsg; 62 } 63 64 //POST方式發送HTTP請求 65 public static void main(String[] args) { 66 String json = "{\"PubInfo\": {\"clinet\": \"10.70.128.132\",\"company\": \"月月鳥0820\"},\"Request\": {\"strBillId\": \"18221075148\",\"strCcsOpId\": \"1234\"}}"; 67 String url = "http://localhost:8080/UpDown/postHttpServer"; 68 String outPackage = null; 69 outPackage = postHttpReq(json, url); 70 System.out.println("客戶端日志----POST方式調用HTTP,請求報文為:" + json); 71 System.out 72 .println("\nauthor<pantp>===========客戶端日志----POST方式調用HTTP服務,HTTP服務端響應報文如下:=============\n"); 73 74 System.out.println(outPackage); 75 76 System.out 77 .println("\nauthor<pantp>================================================================\n"); 78 } 79 80 }
3.測試結果
部署項目,啟動tomc,運行客戶端代碼中的main方式
客戶端的測試結果如下:
服務端的測試結果如下: