HttpClient

参考:
[url]http://wangxinchun.iteye.com/blog/2136254[/url]
[url]http://wangxinchun.iteye.com/blog/2156660[/url]
[url]http://wangxinchun.iteye.com/blog/2166837[/url]


一个大的系统为了解耦,一般进行分离为多个系统,多个系统之间必然存在交互。
http协议是此种交互中最常用的,简单方便,效率也不错。

基于java语言的 apache 的开源项目HttpClient 是比较常用的工具。

我发现这个工具,很多时候大家用的并不好:
1、http连接池PoolingClientConnectionManager,项目中几乎不用。
2、DefaultHttpClient对象,每次调用都新生成,它是线程安全的,一个项目中一个即可。

下面我分享下我先前公司的使用案例:(仅供参考)
pom配置:

<dependency>
<groupId>org.apache.httpcomponents</groupId>
<artifactId>httpclient</artifactId>
<version>4.2</version>
</dependency>
<dependency>
<groupId>org.apache.httpcomponents</groupId>
<artifactId>httpmime</artifactId>
<version>4.2</version>
</dependency>


java 代码:

package com.***.framework.res.api.util;

import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.apache.http.Header;
import org.apache.http.HeaderElement;
import org.apache.http.HttpEntity;
import org.apache.http.HttpRequest;
import org.apache.http.HttpRequestInterceptor;
import org.apache.http.HttpResponse;
import org.apache.http.HttpResponseInterceptor;
import org.apache.http.NameValuePair;
import org.apache.http.client.HttpClient;
import org.apache.http.client.entity.GzipDecompressingEntity;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpDelete;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.methods.HttpPut;
import org.apache.http.entity.ContentType;
import org.apache.http.entity.StringEntity;
import org.apache.http.entity.mime.MultipartEntity;
import org.apache.http.entity.mime.content.ByteArrayBody;
import org.apache.http.entity.mime.content.ContentBody;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.protocol.HttpContext;
import org.apache.http.util.EntityUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.***.framework.res.api.exception.DajieHttpRequestException;

/**
* http请求工具类 get,post
*
* @author xinchun.wang
*
*/
public class HttpClientUtil {
private static Logger logger = LoggerFactory.getLogger(HttpClientUtil.class);
private static final String HEADER_ACCEPT_ENCODING = "Accept-Encoding";
private static final String ENCODING_GZIP = "gzip";
private HttpClient httpclient;


/**
* 通过post提交方式获取url指定的资源和数据
*
* @param url
* @return
* @throws DajieHttpRequestException
*/
public String postData(String url) throws DajieHttpRequestException {
return postData(url, null);
}

/**
* 通过post提交方式获取url指定的资源和数据
*
* @param url
* @param nameValuePairs
* 请求参数
* @return
* @throws DajieHttpRequestException
*/
public String postData(String url, List<NameValuePair> nameValuePairs)
throws DajieHttpRequestException {
return postData(url, nameValuePairs, null);
}

/**
* 通过post提交方式获取url指定的资源和数据
*
* @param url
* @param nameValuePairs
* 请求参数
* @param headers
* 请求header参数
* @return
* @throws DajieHttpRequestException
*/
public String postData(String url,
List<NameValuePair> nameValuePairs, Map<String, String> headers)
throws DajieHttpRequestException {
long start = System.currentTimeMillis();
HttpPost httpPost = new HttpPost(url);
try {
if (headers != null && headers.size() > 0) {
Set<Map.Entry<String, String>> set = headers.entrySet();
for (Iterator<Map.Entry<String, String>> it = set.iterator(); it
.hasNext();) {
Map.Entry<String, String> header = it.next();
if (header != null) {
httpPost.setHeader(header.getKey(), header.getValue());
}
}
}
if (nameValuePairs != null && nameValuePairs.size() > 0) {
httpPost.setEntity(new UrlEncodedFormEntity(nameValuePairs,"UTF-8"));
}

HttpResponse response = httpclient.execute(httpPost);
HttpEntity entity = response.getEntity();
if(entity == null){
return null;
}
String info = EntityUtils.toString(entity, "UTF-8");
return info;
} catch (Exception e) {
logger.debug("postData Exception url: {}", url, e);
throw new DajieHttpRequestException(url
+ "dajie postData exception:", e);
} finally {
httpPost.releaseConnection();
long interval = System.currentTimeMillis() - start;
logger.debug("{} 请求耗时:{} ", url, interval);
}
}

/**
* 通过ContentType 为json的格式进行http传输
* @param url 远程url
* @param content 传输内容
* @return
* @throws DajieHttpRequestException
*/
public String postJSONData(String url, String content) throws DajieHttpRequestException {
long start = System.currentTimeMillis();
HttpPost httpPost = new HttpPost(url);
try {
if (content != null && content.length() > 0) {
httpPost.setEntity(new StringEntity(content,ContentType.APPLICATION_JSON));
}
HttpResponse response = httpclient.execute(httpPost);
HttpEntity entity = response.getEntity();
if(entity == null){
return null;
}
String info = EntityUtils.toString(entity, "UTF-8");
return info;
} catch (Exception e) {
logger.debug("postData Exception url: {}", url, e);
throw new DajieHttpRequestException(url
+ "dajie postDataByJson exception:", e);
} finally {
httpPost.releaseConnection();
long interval = System.currentTimeMillis() - start;
logger.debug("{} 请求耗时:{} ", url, interval);
}
}

/**
* 通过get方法获取url资源的数据
*
* @param url
* 服务器地址
* @return 返回响应的文本,如果请求发生异常,抛出DajieHttpRequestException
* @throws DajieHttpRequestException
*/
public String getData(String url) throws DajieHttpRequestException {
return getData(url, null);
}

/**
* 带header的get请求
*
* @param url
* 服务器地址
* @param headers
* 添加的请求header信息
* @return 返回服务器响应的文本,出错抛出DajieHttpRequestException异常
* @throws DajieHttpRequestException
*/
public String getData(String url, Map<String, String> headers)
throws DajieHttpRequestException {
long start = System.currentTimeMillis();
HttpGet httpGet = new HttpGet(url);
if (headers != null && headers.size() > 0) {
Set<Map.Entry<String, String>> set = headers.entrySet();
for (Iterator<Map.Entry<String, String>> it = set.iterator(); it
.hasNext();) {
Map.Entry<String, String> header = it.next();
if (header != null) {
httpGet.setHeader(header.getKey(), header.getValue());
}
}
}
try {
HttpResponse response = httpclient.execute(httpGet);
HttpEntity entity = response.getEntity();
if(entity == null){
return null;
}
String info = EntityUtils.toString(entity, "UTF-8");
return info;
} catch (Exception e) {
logger.debug("getData Exception url: {}", url, e);
throw new DajieHttpRequestException(url
+ "dajie getData exception:", e);
} finally {
httpGet.releaseConnection();
long interval = System.currentTimeMillis() - start;
logger.debug("{} 请求耗时:{} ", url, interval);
}
}

/**
* 对httpclient 做压缩处理和解压缩处理
*
* @param httpclient
*/
public void initClient() {
((DefaultHttpClient)httpclient).addRequestInterceptor(new HttpRequestInterceptor() {
@Override
public void process(HttpRequest request, HttpContext context) {
if (!request.containsHeader(HEADER_ACCEPT_ENCODING)) {
request.addHeader(HEADER_ACCEPT_ENCODING, ENCODING_GZIP);
}
}
});

((DefaultHttpClient)httpclient).addResponseInterceptor(new HttpResponseInterceptor() {
@Override
public void process(HttpResponse response, HttpContext context) {
final HttpEntity entity = response.getEntity();
if(entity == null){
return;
}
final Header encoding = entity.getContentEncoding();
if (encoding != null) {
for (HeaderElement element : encoding.getElements()) {
if (element.getName().equalsIgnoreCase(ENCODING_GZIP)) {
response.setEntity(new GzipDecompressingEntity(
response.getEntity()));
break;
}
}
}
}
});
}

/**
* 关闭客户端
*/
public void destroyClient(){
httpclient.getConnectionManager().shutdown();
}
/**
* post方式处理文件和图片上传
*
* @param url
* 服务器地址
* @param data
* byte数组数据
* @param fileName
* 文件名
* @return 返回服务器响应信息,否则抛出DajieHttpRequestException异常
* @throws DajieHttpRequestException
*/
public String postMultipartData(String url, byte[] data,
String fileName) throws DajieHttpRequestException {
long start = System.currentTimeMillis();
HttpPost httpPost = new HttpPost(url);
try {
if (data != null && data.length > 0) {
MultipartEntity reqEntity = new MultipartEntity();
ContentBody contentBody = new ByteArrayBody(data, fileName);
reqEntity.addPart("file", contentBody);
httpPost.setEntity(reqEntity);
}
HttpResponse response = httpclient.execute(httpPost);
HttpEntity entity = response.getEntity();
String info = EntityUtils.toString(entity, "UTF-8");
return info;
} catch (Exception e) {
logger.debug("postMultipartData Exception url: {}", url, e);
throw new DajieHttpRequestException(url
+ "dajie postMultipartData exception:", e);
} finally {
httpPost.releaseConnection();
long interval = System.currentTimeMillis() - start;
logger.debug("{} 请求耗时:{} ", url, interval);
}
}

/**
* put 方式提交数据
*
* @param url
* :服务器地址
* @param nameValuePairs
* :参数
* @return 返回 服务器返回的文本信息,报错会抛出异常
* @throws DajieHttpRequestException
*/
public String putData(String url, List<NameValuePair> nameValuePairs)
throws DajieHttpRequestException {
long start = System.currentTimeMillis();
HttpPut httpPut = new HttpPut(url);

try {
if (nameValuePairs != null && nameValuePairs.size() > 0) {
httpPut.setEntity(new UrlEncodedFormEntity(nameValuePairs,
"UTF-8"));
}
HttpResponse response = httpclient.execute(httpPut);
HttpEntity entity = response.getEntity();
if(entity == null){
return null;
}
String info = EntityUtils.toString(entity, "UTF-8");
return info;
} catch (Exception e) {
logger.debug("putData Exception url:{}", url, e);
throw new DajieHttpRequestException(url
+ "dajie putData exception:", e);
} finally {
httpPut.releaseConnection();
long interval = System.currentTimeMillis() - start;
logger.debug("{} 请求耗时:{} ", url, interval);
}
}

/**
* delete 方式提交数据
*
* @param url
* 服务器地址
* @return 返回 服务器返回的文本信息,报错会抛出异常
* @throws DajieHttpRequestException
*/
public String deleteData(String url)
throws DajieHttpRequestException {
return deleteData(url, null);
}

/**
* delete 方式提交数据
*
* @param url
* 服务器地址
* @return 返回 服务器返回的文本信息,报错会抛出异常
*/
public String deleteData(String url, Map<String, String> headers)
throws DajieHttpRequestException {
long start = System.currentTimeMillis();
HttpDelete httpDelete = new HttpDelete(url);

if (headers != null && headers.size() > 0) {
Set<Map.Entry<String, String>> set = headers.entrySet();
for (Iterator<Map.Entry<String, String>> it = set.iterator(); it
.hasNext();) {
Map.Entry<String, String> header = it.next();
if (header != null) {
httpDelete.setHeader(header.getKey(), header.getValue());
}
}
}
try {
HttpResponse response = httpclient.execute(httpDelete);
HttpEntity entity = response.getEntity();
String info = EntityUtils.toString(entity, "UTF-8");
return info;
} catch (Exception e) {
logger.debug("putData Exception url {} ", url, e);
throw new DajieHttpRequestException(url
+ "dajie deleteDate exception:", e);
} finally {
httpDelete.releaseConnection();
long interval = System.currentTimeMillis() - start;
logger.debug("{} 请求耗时:{} ", url, interval);
}
}

/**
* 下载媒体资源
* @param url
* @return
* @throws DajieHttpRequestException
*/
public byte[] getMultipartData(String url) throws DajieHttpRequestException{
long start = System.currentTimeMillis();
HttpGet httpGet = new HttpGet(url);
try {
HttpResponse response = httpclient.execute(httpGet);
byte[] result = EntityUtils.toByteArray(response.getEntity());
return result;
}catch(Exception e){
logger.debug("putData Exception url {} ", url, e);
throw new DajieHttpRequestException(url+ "dajie getMultipartData exception:", e);
}finally{
httpGet.releaseConnection();
long interval = System.currentTimeMillis() - start;
logger.debug("{} 请求耗时:{} ", url, interval);
}
}

public void setHttpclient(HttpClient httpclient) {
this.httpclient = httpclient;
}
}



spring 配置:

<bean id="connManager" class="org.apache.http.impl.conn.PoolingClientConnectionManager">
<property name="maxTotal" value="#{internalConfiguration.maxTotal}"/>
<property name="defaultMaxPerRoute" value="#{internalConfiguration.defaultMaxPerRoute}"/>
</bean>

<bean id="httpclient" class="org.apache.http.impl.client.DefaultHttpClient">
<constructor-arg>
<ref bean="connManager"/>
</constructor-arg>
</bean>

<bean id="httpClientUtil" class="com.***.framework.res.api.util.HttpClientUtil" init-method="initClient" destroy-method="destroyClient">
<property name="httpclient" ref="httpclient"/>
</bean>


注意:这个配置很重要,尤其 是参数:maxTotal 和 defaultMaxPerRoute 。
defaultMaxPerRoute :对一个路由(url)地址,最多同时存在几个连接。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值