HttpClientUtil工具類,傳送get請求和post請求
阿新 • • 發佈:2019-02-01
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.HttpURLConnection;
import java.net.URL;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLException;
import javax.net.ssl.SSLSession;
import javax.net.ssl.SSLSocket;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.http.Consts;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.NameValuePair;
import org.apache.http.ParseException;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.utils.URLEncodedUtils;
import org.apache.http.conn.scheme.Scheme;
import org.apache.http.conn.ssl.SSLSocketFactory;
import org.apache.http.conn.ssl.X509HostnameVerifier;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.protocol.HTTP;
import org.apache.http.util.EntityUtils;
/**
* 封裝了一些採用HttpClient傳送HTTP請求的方法
* @see 本工具所採用的是最新的HttpComponents-Client-4.2.1
*/
public class HttpClientUtil {
private HttpClientUtil(){}
private static Log logger = LogFactory.getLog(HttpClientUtil.class);
/**
* 傳送HTTP_GET請求
* @see 該方法會自動關閉連線,釋放資源
* @param requestURL 請求地址(含引數)
* @param decodeCharset 解碼字符集,解析響應資料時用之,其為null時預設採用UTF-8解碼
* @return 遠端主機響應正文
*/
public static String sendGetRequest(String reqURL, String decodeCharset){
long responseLength = 0; //響應長度
String responseContent = null; //響應內容
HttpClient httpClient = new DefaultHttpClient(); //建立預設的httpClient例項
HttpGet httpGet = new HttpGet(reqURL); //建立org.apache.http.client.methods.HttpGet
try{
HttpResponse response = httpClient.execute(httpGet); //執行GET請求
HttpEntity entity = response.getEntity(); //獲取響應實體
if(null != entity){
responseLength = entity.getContentLength();
responseContent = EntityUtils.toString(entity, decodeCharset==null ? "UTF-8" : decodeCharset);
EntityUtils.consume(entity); //Consume response content
}
System.out.println("請求地址: " + httpGet.getURI());
System.out.println("響應狀態: " + response.getStatusLine());
System.out.println("響應長度: " + responseLength);
System.out.println("響應內容: " + responseContent);
}catch(ClientProtocolException e){
logger.debug("該異常通常是協議錯誤導致,比如構造HttpGet物件時傳入的協議不對(將'http'寫成'htp')或者伺服器端返回的內容不符合HTTP協議要求等,堆疊資訊如下", e);
}catch(ParseException e){
logger.debug(e.getMessage(), e);
}catch(IOException e){
logger.debug("該異常通常是網路原因引起的,如HTTP伺服器未啟動等,堆疊資訊如下", e);
}finally{
httpClient.getConnectionManager().shutdown(); //關閉連線,釋放資源
}
return responseContent;
}
/**
* 傳送HTTP_POST請求
* @see 該方法為sendPostRequest(String,String,boolean,String,String)
的簡化方法
* @see 該方法在對請求資料的編碼和響應資料的解碼時,所採用的字符集均為UTF-8
* @see 當isEncoder=true
時,其會自動對sendData
中的[中文][|][ ]等特殊字元進行URLEncoder.encode(string,"UTF-8")
* @param isEncoder 用於指明請求資料是否需要UTF-8編碼,true為需要
*/
public static String sendPostRequest(String reqURL, String sendData, boolean isEncoder){
return sendPostRequest(reqURL, sendData, isEncoder, null, null);
}
/**
* 傳送HTTP_POST請求
* @see 該方法會自動關閉連線,釋放資源
* @see 當isEncoder=true
時,其會自動對sendData
中的[中文][|][ ]等特殊字元進行URLEncoder.encode(string,encodeCharset)
* @param reqURL 請求地址
* @param sendData 請求引數,若有多個引數則應拼接成param11=value11&22=value22&33=value33的形式後,傳入該引數中
* @param isEncoder 請求資料是否需要encodeCharset編碼,true為需要
* @param encodeCharset 編碼字符集,編碼請求資料時用之,其為null時預設採用UTF-8解碼
* @param decodeCharset 解碼字符集,解析響應資料時用之,其為null時預設採用UTF-8解碼
* @return 遠端主機響應正文
*/
public static String sendPostRequest(String reqURL, String sendData, boolean isEncoder, String encodeCharset, String decodeCharset){
String responseContent = null;
HttpClient httpClient = new DefaultHttpClient();
HttpPost httpPost = new HttpPost(reqURL);
//httpPost.setHeader(HTTP.CONTENT_TYPE, "application/x-www-form-urlencoded; charset=UTF-8");
httpPost.setHeader(HTTP.CONTENT_TYPE, "application/x-www-form-urlencoded");
try{
if(isEncoder){
List formParams = new ArrayList();
for(String str : sendData.split("&")){
formParams.add(new BasicNameValuePair(str.substring(0,str.indexOf("=")), str.substring(str.indexOf("=")+1)));
}
httpPost.setEntity(new StringEntity(URLEncodedUtils.format(formParams, encodeCharset==null ? "UTF-8" : encodeCharset)));
}else{
httpPost.setEntity(new StringEntity(sendData));
}
HttpResponse response = httpClient.execute(httpPost);
HttpEntity entity = response.getEntity();
if (null != entity) {
responseContent = EntityUtils.toString(entity, decodeCharset==null ? "UTF-8" : decodeCharset);
EntityUtils.consume(entity);
}
}catch(Exception e){
logger.debug("與[" + reqURL + "]通訊過程中發生異常,堆疊資訊如下", e);
}finally{
httpClient.getConnectionManager().shutdown();
}
return responseContent;
}
/**
* 傳送HTTP_POST請求
* @see 該方法會自動關閉連線,釋放資源
* @see 該方法會自動對params
中的[中文][|][ ]等特殊字元進行URLEncoder.encode(string,encodeCharset)
* @param reqURL 請求地址
* @param params 請求引數
* @param encodeCharset 編碼字符集,編碼請求資料時用之,其為null時預設採用UTF-8解碼
* @param decodeCharset 解碼字符集,解析響應資料時用之,其為null時預設採用UTF-8解碼
* @return 遠端主機響應正文
*/
public static String sendPostRequest(String reqURL, Map params, String encodeCharset, String decodeCharset){
String responseContent = null;
HttpClient httpClient = new DefaultHttpClient();
HttpPost httpPost = new HttpPost(reqURL);
List formParams = new ArrayList(); //建立引數佇列
for(Map.Entry entry : params.entrySet()){
formParams.add(new BasicNameValuePair(entry.getKey(), entry.getValue()));
}
try{
httpPost.setEntity(new UrlEncodedFormEntity(formParams, encodeCharset==null ? "UTF-8" : encodeCharset));
HttpResponse response = httpClient.execute(httpPost);
HttpEntity entity = response.getEntity();
if (null != entity) {
responseContent = EntityUtils.toString(entity, decodeCharset==null ? "UTF-8" : decodeCharset);
EntityUtils.consume(entity);
}
}catch(Exception e){
logger.debug("與[" + reqURL + "]通訊過程中發生異常,堆疊資訊如下", e);
}finally{
httpClient.getConnectionManager().shutdown();
}
return responseContent;
}
/**
* 傳送HTTPS_POST請求
* @see 該方法為sendPostSSLRequest(String,Map,String,String)
方法的簡化方法
* @see 該方法在對請求資料的編碼和響應資料的解碼時,所採用的字符集均為UTF-8
* @see 該方法會自動對params
中的[中文][|][ ]等特殊字元進行URLEncoder.encode(string,"UTF-8")
*/
public static String sendPostSSLRequest(String reqURL, Map params){
return sendPostSSLRequest(reqURL, params, null, null);
}
/**
* 傳送HTTPS_POST請求
* @see 該方法會自動關閉連線,釋放資源
* @see 該方法會自動對params
中的[中文][|][ ]等特殊字元進行URLEncoder.encode(string,encodeCharset)
* @param reqURL 請求地址
* @param params 請求引數
* @param encodeCharset 編碼字符集,編碼請求資料時用之,其為null時預設採用UTF-8解碼
* @param decodeCharset 解碼字符集,解析響應資料時用之,其為null時預設採用UTF-8解碼
* @return 遠端主機響應正文
*/
public static String sendPostSSLRequest(String reqURL, Map params, String encodeCharset, String decodeCharset){
String responseContent = "";
HttpClient httpClient = new DefaultHttpClient();
X509TrustManager xtm = new X509TrustManager(){
public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
public X509Certificate[] getAcceptedIssuers() {return null;}
};
try {
SSLContext ctx = SSLContext.getInstance("TLS");
ctx.init(null, new TrustManager[]{xtm}, null);
SSLSocketFactory socketFactory = new SSLSocketFactory(ctx);
httpClient.getConnectionManager().getSchemeRegistry().register(new Scheme("https", 443, socketFactory));
HttpPost httpPost = new HttpPost(reqURL);
List formParams = new ArrayList();
for(Map.Entry entry : params.entrySet()){
formParams.add(new BasicNameValuePair(entry.getKey(), entry.getValue()));
}
httpPost.setEntity(new UrlEncodedFormEntity(formParams, encodeCharset==null ? "UTF-8" : encodeCharset));
HttpResponse response = httpClient.execute(httpPost);
HttpEntity entity = response.getEntity();
if (null != entity) {
responseContent = EntityUtils.toString(entity, decodeCharset==null ? "UTF-8" : decodeCharset);
EntityUtils.consume(entity);
}
} catch (Exception e) {
logger.debug("與[" + reqURL + "]通訊過程中發生異常,堆疊資訊為", e);
} finally {
httpClient.getConnectionManager().shutdown();
}
return responseContent;
}
/**
* 傳送HTTP_POST請求
* @see 若傳送的params
中含有中文,記得按照雙方約定的字符集將中文URLEncoder.encode(string,encodeCharset)
* @see 本方法預設的連線超時時間為30秒,預設的讀取超時時間為30秒
* @param reqURL 請求地址
* @param params 傳送到遠端主機的正文資料,其資料型別為java.util.Map
* @return 遠端主機響應正文`HTTP狀態碼,如"SUCCESS`200"
若通訊過程中發生異常則返回"Failed`HTTP狀態碼",如"Failed`500"
*/
public static String sendPostRequestByJava(String reqURL, Map params){
StringBuilder sendData = new StringBuilder();
for(Map.Entry entry : params.entrySet()){
sendData.append(entry.getKey()).append("=").append(entry.getValue()).append("&");
}
if(sendData.length() > 0){
sendData.setLength(sendData.length() - 1); //刪除最後一個&符號
}
return sendPostRequestByJava(reqURL, sendData.toString());
}
/**
* 傳送HTTP_POST請求
* @see 若傳送的sendData
中含有中文,記得按照雙方約定的字符集將中文URLEncoder.encode(string,encodeCharset)
* @see 本方法預設的連線超時時間為30秒,預設的讀取超時時間為30秒
* @param reqURL 請求地址
* @param sendData 傳送到遠端主機的正文資料
* @return 遠端主機響應正文`HTTP狀態碼,如"SUCCESS`200"
若通訊過程中發生異常則返回"Failed`HTTP狀態碼",如"Failed`500"
*/
public static String sendPostRequestByJava(String reqURL, String sendData){
HttpURLConnection httpURLConnection = null;
OutputStream out = null; //寫
InputStream in = null; //讀
int httpStatusCode = 0; //遠端主機響應的HTTP狀態碼
try{
URL sendUrl = new URL(reqURL);
httpURLConnection = (HttpURLConnection)sendUrl.openConnection();
httpURLConnection.setRequestMethod("POST");
httpURLConnection.setDoOutput(true); //指示應用程式要將資料寫入URL連線,其值預設為false
httpURLConnection.setUseCaches(false);
httpURLConnection.setConnectTimeout(30000); //30秒連線超時
httpURLConnection.setReadTimeout(30000); //30秒讀取超時
out = httpURLConnection.getOutputStream();
out.write(sendData.toString().getBytes());
//清空緩衝區,傳送資料
out.flush();
//獲取HTTP狀態碼
httpStatusCode = httpURLConnection.getResponseCode();
in = httpURLConnection.getInputStream();
byte[] byteDatas = new byte[in.available()];
in.read(byteDatas);
return new String(byteDatas) + "`" + httpStatusCode;
}catch(Exception e){
logger.debug(e.getMessage());
return "Failed`" + httpStatusCode;
}finally{
if(out != null){
try{
out.close();
}catch (Exception e){
logger.debug("關閉輸出流時發生異常,堆疊資訊如下", e);
}
}
if(in != null){
try{
in.close();
}catch(Exception e){
logger.debug("關閉輸入流時發生異常,堆疊資訊如下", e);
}
}
if(httpURLConnection != null){
httpURLConnection.disconnect();
httpURLConnection = null;
}
}
}
/**
* https posp請求,可以繞過證書校驗
* @param url
* @param params
* @return
*/
public static final String sendHttpsRequestByPost(String url, Map params) {
String responseContent = null;
HttpClient httpClient = new DefaultHttpClient();
//建立TrustManager
X509TrustManager xtm = new X509TrustManager() {
public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {}
public X509Certificate[] getAcceptedIssuers() {
return null;
}
};
//這個好像是HOST驗證
X509HostnameVerifier hostnameVerifier = new X509HostnameVerifier() {
public boolean verify(String arg0, SSLSession arg1) {
return true;
}
public void verify(String arg0, SSLSocket arg1) throws IOException {}
public void verify(String arg0, String[] arg1, String[] arg2) throws SSLException {}
public void verify(String arg0, X509Certificate arg1) throws SSLException {}
};
try {
//TLS1.0與SSL3.0基本上沒有太大的差別,可粗略理解為TLS是SSL的繼承者,但它們使用的是相同的SSLContext
SSLContext ctx = SSLContext.getInstance("TLS");
//使用TrustManager來初始化該上下文,TrustManager只是被SSL的Socket所使用
ctx.init(null, new TrustManager[] { xtm }, null);
//建立SSLSocketFactory
SSLSocketFactory socketFactory = new SSLSocketFactory(ctx);
socketFactory.setHostnameVerifier(hostnameVerifier);
//通過SchemeRegistry將SSLSocketFactory註冊到我們的HttpClient上
httpClient.getConnectionManager().getSchemeRegistry().register(new Scheme("https", socketFactory, 443));
HttpPost httpPost = new HttpPost(url);
List formParams = new ArrayList(); // 構建POST請求的表單引數
for (Map.Entry entry : params.entrySet()) {
formParams.add(new BasicNameValuePair(entry.getKey(), entry.getValue()));
}
httpPost.setEntity(new UrlEncodedFormEntity(formParams, "UTF-8"));
HttpResponse response = httpClient.execute(httpPost);
HttpEntity entity = response.getEntity(); // 獲取響應實體
if (entity != null) {
responseContent = EntityUtils.toString(entity, "UTF-8");
}
} catch (KeyManagementException e) {
e.printStackTrace();
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} catch (ClientProtocolException e) {
e.printStackTrace();
} catch (ParseException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
// 關閉連線,釋放資源
httpClient.getConnectionManager().shutdown();
}
return responseContent;
}
/**
* 傳送HTTP_POST請求,json格式資料
* @param url
* @param body
* @return
* @throws Exception
*/
public static String sendPostByJson(String url, String body) throws Exception {
CloseableHttpClient httpclient = HttpClients.custom().build();
HttpPost post = null;
String resData = null;
CloseableHttpResponse result = null;
try {
post = new HttpPost(url);
HttpEntity entity2 = new StringEntity(body, Consts.UTF_8);
post.setConfig(RequestConfig.custom().setConnectTimeout(30000).setSocketTimeout(30000).build());
post.setHeader("Content-Type", "application/json");
post.setEntity(entity2);
result = httpclient.execute(post);
if (HttpStatus.SC_OK == result.getStatusLine().getStatusCode()) {
resData = EntityUtils.toString(result.getEntity());
}
} finally {
if (result != null) {
result.close();
}
if (post != null) {
post.releaseConnection();
}
httpclient.close();
}
return resData;
}
}