一、背景

业务开发中,经常会遇到通过http/https向下游服务发送请求。每次都要重复造轮子写HttpClient的逻辑,而且性能、功能参差不齐。这里分享一个高性能的、带连接池的通用Http客户端工具。

请尊重作者劳动成果,转载请标明原文链接:https://www.cnblogs.com/waterystone/p/11551280.html

二、特点

  • 基于apache的高性能Http客户端org.apache.http.client.HttpClient;
  • 连接池的最大连接数默认是20,可通过系统变量-Dadu.common.http.max.total=200指定;
  • 连接池的每个路由的最大连接数默认是2,可通过系统变量-Dadu.common.http.max.per.route=10指定;
  • 可设置超时,通过HttpOptions进行设置;
  • 可重试,通过HttpOptions进行设置。

三、源码

参考:https://github.com/waterystone/adu-test/blob/master/src/main/java/com/adu/utils/HttpClientUtil.java

 package com.adu.utils;

 import com.adu.Constants;
import com.adu.handler.HttpRequestRetryHandler;
import com.adu.model.HttpOptions;
import com.adu.model.HttpRequest;
import com.adu.model.HttpResponse;
import org.apache.http.Header;
import org.apache.http.HttpEntity;
import org.apache.http.NameValuePair;
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.methods.HttpRequestBase;
import org.apache.http.client.utils.URIBuilder;
import org.apache.http.client.utils.URLEncodedUtils;
import org.apache.http.config.Registry;
import org.apache.http.config.RegistryBuilder;
import org.apache.http.conn.socket.ConnectionSocketFactory;
import org.apache.http.conn.socket.PlainConnectionSocketFactory;
import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
import org.apache.http.entity.StringEntity;
import org.apache.http.entity.mime.MultipartEntityBuilder;
import org.apache.http.entity.mime.content.ContentBody;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.impl.conn.PoolingHttpClientConnectionManager;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.util.EntityUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import javax.net.ssl.SSLContext;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.stream.Collectors; /**
* 带有连接池的Http客户端工具类。具有如下特点:
* <ol>
* <li>基于apache的高性能Http客户端{@link org.apache.http.client.HttpClient};</li>
* <li>连接池的最大连接数默认是20,可通过{@link #init(int, int)}、或者系统变量-Dzzarch.common.http.max.total=200指定;</li>
* <li>连接池的每个路由的最大连接数默认是2,可通过{@link #init(int, int)}、或者系统变量-Dzzarch.common.http.max.per.route=10指定;</li>
* <li>可设置超时,通过{@link HttpOptions}进行设置;</li>
* <li>可重试,通过{@link HttpOptions}进行设置;</li>
* </ol>
*
* @author duyunjie
* @date 2019-09-18 16:33
*/
public class HttpClientUtil {
private static final Logger logger = LoggerFactory.getLogger(HttpClientUtil.class); /**
* HttpClient 连接池
*/
private static PoolingHttpClientConnectionManager CONNECTION_MANAGER = buildPoolingHttpClientConnectionManager(null, null); /**
* @param maxTotal 连接池的最大连接数,默认为20。
* @param maxPerRoute 连接池的每个路由的最大连接数,默认为2。
*/
public static void init(int maxTotal, int maxPerRoute) {
CONNECTION_MANAGER = buildPoolingHttpClientConnectionManager(maxTotal, maxPerRoute);
} public static HttpResponse httpGet(HttpRequest httpRequest) throws Exception {
return httpGet(httpRequest, null);
} /**
* 发送 HTTP GET请求
*
* @param httpRequest 请求参数,如url,header等。
* @param httpOptions 配置参数,如重试次数、超时时间等。
* @return
* @throws Exception
*/
public static HttpResponse httpGet(HttpRequest httpRequest, HttpOptions httpOptions) throws Exception {
// 装载请求地址和参数
URIBuilder ub = new URIBuilder(httpRequest.getUrl()); // 转换请求参数
List<NameValuePair> pairs = convertParams2NVPS(httpRequest.getParams());
if (!pairs.isEmpty()) {
ub.setParameters(pairs);
}
HttpGet httpGet = new HttpGet(ub.build()); // 设置请求头
if (Objects.nonNull(httpRequest.getHeaders())) {
for (Map.Entry<String, String> header : httpRequest.getHeaders().entrySet()) {
httpGet.addHeader(header.getKey(), String.valueOf(header.getValue()));
}
} return doHttp(httpGet, httpOptions);
} public static HttpResponse httpPost(HttpRequest httpRequest) throws Exception {
return httpPost(httpRequest, null);
} /**
* 发送 HTTP POST请求
*
* @param httpRequest 请求参数
* @param httpOptions 配置参数
* @return
* @throws Exception
*/
public static HttpResponse httpPost(HttpRequest httpRequest, HttpOptions httpOptions) throws Exception {
HttpPost httpPost = new HttpPost(httpRequest.getUrl()); // 转换请求参数
List<NameValuePair> pairs = convertParams2NVPS(httpRequest.getParams());
if (!pairs.isEmpty()) {
httpPost.setEntity(new UrlEncodedFormEntity(pairs, StandardCharsets.UTF_8.name()));
} // 设置请求头
if (Objects.nonNull(httpRequest.getHeaders())) {
for (Map.Entry<String, String> header : httpRequest.getHeaders().entrySet()) {
httpPost.addHeader(header.getKey(), String.valueOf(header.getValue()));
}
} return doHttp(httpPost, httpOptions);
} /**
* 发送 HTTP POST请求,参数格式JSON
* <p>请求参数是JSON格式,数据编码是UTF-8</p>
*
* @param url
* @param param
* @return
* @throws Exception
*/
public static HttpResponse httpPostJson(String url, String param, HttpOptions httpOptions) throws Exception {
HttpPost httpPost = new HttpPost(url); // 设置请求头
httpPost.addHeader("Content-Type", "application/json; charset=UTF-8"); // 设置请求参数
httpPost.setEntity(new StringEntity(param, StandardCharsets.UTF_8.name())); return doHttp(httpPost, httpOptions);
} /**
* 发送 HTTP POST请求,参数格式XML
* <p>请求参数是XML格式,数据编码是UTF-8</p>
*
* @param url
* @param param
* @return
* @throws Exception
*/
public static HttpResponse httpPostXml(String url, String param, HttpOptions httpOptions) throws Exception {
HttpPost httpPost = new HttpPost(url); // 设置请求头
httpPost.addHeader("Content-Type", "application/xml; charset=UTF-8"); // 设置请求参数
httpPost.setEntity(new StringEntity(param, StandardCharsets.UTF_8.name())); return doHttp(httpPost, httpOptions);
} /**
* 通过post发送multipart信息。
*
* @param url
* @param multiparts
* @param httpOptions
* @return
* @throws Exception
*/
public static HttpResponse httpPostMultipart(String url, Map<String, ContentBody> multiparts, HttpOptions httpOptions) throws Exception {
HttpPost httpPost = new HttpPost(url); // 设置Multipart
if (Objects.nonNull(multiparts)) {
MultipartEntityBuilder multipartEntityBuilder = MultipartEntityBuilder.create();
for (Map.Entry<String, ContentBody> multipartEntry : multiparts.entrySet()) {
multipartEntityBuilder.addPart(multipartEntry.getKey(), multipartEntry.getValue());
} httpPost.setEntity(multipartEntityBuilder.build());
} return doHttp(httpPost, httpOptions);
} /**
* 转换请求参数,将Map键值对拼接成QueryString字符串
*
* @param params
* @return
*/
public static String convertParams2QueryStr(Map<String, ?> params) {
List<NameValuePair> pairs = convertParams2NVPS(params); return URLEncodedUtils.format(pairs, StandardCharsets.UTF_8.name());
} /**
* 转换请求参数
*
* @param params
* @return
*/
public static List<NameValuePair> convertParams2NVPS(Map<String, ?> params) {
if (Objects.isNull(params)) {
return new ArrayList<>();
} return params.entrySet().stream().map(param -> new BasicNameValuePair(param.getKey(), String.valueOf(param.getValue()))).collect(Collectors.toList());
} /**
* 发送 HTTP 请求
*
* @param request
* @return
* @throws Exception
*/
private static HttpResponse doHttp(HttpRequestBase request, HttpOptions httpOptions) throws Exception {
if (Objects.isNull(httpOptions)) {//如果为空,则用默认的。
httpOptions = HttpOptions.DEFAULT_HTTP_OPTION;
}
// 设置超时时间
if (Objects.nonNull(httpOptions.getTimeoutMs())) {
request.setConfig(RequestConfig.custom().setSocketTimeout(httpOptions.getTimeoutMs()).build());
} //设置重试策略
HttpRequestRetryHandler httpRequestRetryHandler = null;
if (Objects.nonNull(httpOptions.getRetryCount())) {
httpRequestRetryHandler = new HttpRequestRetryHandler(httpOptions.getRetryCount());
} // 通过连接池获取连接对象
CloseableHttpClient httpClient = HttpClients.custom().setConnectionManager(CONNECTION_MANAGER).setRetryHandler(httpRequestRetryHandler).build();
return doRequest(httpClient, request); } /**
* 处理Http/Https请求,并返回请求结果
* <p>注:默认请求编码方式 UTF-8</p>
*
* @param httpClient
* @param request
* @return
* @throws Exception
*/
private static HttpResponse doRequest(CloseableHttpClient httpClient, HttpRequestBase request) throws Exception {
HttpResponse res = new HttpResponse();
CloseableHttpResponse response = null;
long start = System.currentTimeMillis(); try {
// 获取请求结果
response = httpClient.execute(request); // 解析请求结果
HttpEntity entity = response.getEntity();
String result = EntityUtils.toString(entity, StandardCharsets.UTF_8.name()); // 转换结果
EntityUtils.consume(entity); // 关闭IO流 //解析返回header
Map<String, String> headers = new HashMap<>(response.getAllHeaders().length);
for (Header header : response.getAllHeaders()) {
headers.put(header.getName(), header.getValue());
} res.setStatusCode(response.getStatusLine().getStatusCode()).setResult(result).setHeaders(headers);
} finally {
if (Objects.nonNull(response)) {
response.close();
}
} long elapsed = System.currentTimeMillis() - start;
logger.debug("op=end_doRequest,request={},res={},elapsed={}", request, res, elapsed);
return res;
} /**
* 初始化连接池
*
* @return
*/
private static PoolingHttpClientConnectionManager buildPoolingHttpClientConnectionManager(Integer maxTotal, Integer maxPerRoute) {
// 初始化连接池,可用于请求HTTP/HTTPS(信任所有证书)
PoolingHttpClientConnectionManager connectionManager = new PoolingHttpClientConnectionManager(getRegistry()); // 整个连接池的最大连接数
String maxTotalProperty = null;
if (Objects.nonNull(maxTotal)) { //首先看有没有在参数中显式指定
connectionManager.setMaxTotal(maxTotal);
} else { //如果没有在参数中显式指定,则再看有没有在系统变量中指定
maxTotalProperty = System.getProperty(Constants.SYSTEM_PROPERTY_KEY_HTTP_MAX_TOTAL);
if (Objects.nonNull(maxTotalProperty)) {
connectionManager.setMaxTotal(Integer.valueOf(maxTotalProperty));
}
} // 每个路由的最大连接数
String maxPerRouteProperty = null;
if (Objects.nonNull(maxPerRoute)) { //首先看有没有在参数中显式指定
connectionManager.setDefaultMaxPerRoute(maxPerRoute);
} else { //如果没有在参数中显式指定,则再看有没有在系统变量中指定
maxPerRouteProperty = System.getProperty(Constants.SYSTEM_PROPERTY_KEY_HTTP_MAX_PER_ROUTE);
if (Objects.nonNull(maxPerRouteProperty)) {
connectionManager.setDefaultMaxPerRoute(Integer.valueOf(maxPerRouteProperty));
}
} logger.info("[ZZARCH_COMMON_SUCCESS_buildPoolingHttpClientConnectionManager]maxTotal={},maxPerRoute={},maxTotalProperty={},maxPerRouteProperty={}", maxTotal, maxPerRoute, maxTotalProperty, maxPerRouteProperty);
return connectionManager;
} /**
* 获取 HTTPClient注册器
*
* @return
* @throws Exception
*/
private static Registry<ConnectionSocketFactory> getRegistry() {
try {
return RegistryBuilder.<ConnectionSocketFactory>create().register("http", new PlainConnectionSocketFactory()).register("https", new SSLConnectionSocketFactory(SSLContext.getDefault())).build();
} catch (Exception e) {
logger.error("[ERROR_getRegistry]", e);
} return null;
}
}

end

带有连接池的Http客户端工具类HttpClientUtil的更多相关文章

  1. 关于jedis2.4以上版本的连接池配置,及工具类

    jedis.propertise 注意以前版本的maxAcitve和maxWait有所改变,JVM根据系统环境变量ServerType中的值 取不同的配置,实现多环境(测试环境.生产环境)集成. re ...

  2. Redis进阶实践之九 独立封装的RedisClient客户端工具类

    一.引言 今天开始有关Redis学习的第九篇文章了,以后肯定会大量系统使用Redis作为缓存介质,为了更好的更好的Redis,自己写了两个工具类,但是这两个工具类,没有提供一致的接口,是为了使用的独立 ...

  3. Redis进阶实践之九 独立封装的RedisClient客户端工具类(转载9)

    Redis进阶实践之九 独立封装的RedisClient客户端工具类 一.引言 今天开始有关Redis学习的第九篇文章了,以后肯定会大量系统使用Redis作为缓存介质,为了更好的更好的Redis,自己 ...

  4. HTTP请求客户端工具类

    1.maven 引入依赖 <dependency> <groupId>commons-httpclient</groupId> <artifactId> ...

  5. 带连接池的netty客户端核心功能实现剖解

    带连接池的netty客户端核心功能实现剖析 带连接池的netty的客户端核心功能实现剖析 本文为原创,转载请注明出处 源码地址: https://github.com/zhangxianwu/ligh ...

  6. Java jdbc 连接oracle之三(封装工具类)

    driver = oracle.jdbc.driver.OracleDriver url = jdbc:oracle:thin:@192.168.10.105:1521:orcl user = LF ...

  7. Jesery客户端工具类

    public class JerseyClientUtil { public static<T> T sendMsg(String url,Object object,Class<T ...

  8. 【JAVAWEB学习笔记】10_JDBC连接池&DBUtils

    使用连接池改造JDBC的工具类: 1.1.1          需求: 传统JDBC的操作,对连接的对象销毁不是特别好.每次创建和销毁连接都是需要花费时间.可以使用连接池优化的程序. * 在程序开始的 ...

  9. C3P0连接池、DBCP连接池

    C3P0连接池: 配置文件:c3p0-config.xml <?xml version="1.0" encoding="UTF-8"?> <c ...

随机推荐

  1. CentOS7安装Oracle 11g数据库

    转载:https://blog.csdn.net/lia17/article/details/82256565 rpm -ivh --force --nodeps *.rpm 强制装 rpm依赖包下载 ...

  2. git使用cherry-pick和revert抢救错误代码提交

    大多数的新手在新接触git时都会出现这样的问题.代码写完了,提交到dev分支进行测试.一高兴忘记切回来,继续在dev分支开发,写完之后提交时猛的发现,我靠,我怎么在dev上面写代码,此时内心必然是一阵 ...

  3. redis之主从同步

    很多企业都没有使用到 Redis 的集群,但是至少都做了主从.有了主从,当 master 挂掉的时候,运维让从库过来接管,服务就可以继续,否则 master 需要经过数据恢复和重启的过程,这就可能会拖 ...

  4. oracle排序子句的特殊写法与ORA-01785错误

    刚刚写的SQL语句在执行的时候报[ORA-01785: ORDER BY item must be the number of a SELECT-list expression]错误,于是自己百度了一 ...

  5. 一个比 AutoMapper 更快的模型映射的组件 Mapster

    下面是官方的性能测试 Demo,感性的也可以去 Github 上下载. 贴出代码目的是如果后期直接从自己的博客中在线看. using System; using System.Collections. ...

  6. YII 项目部署时, 显示空白内容

    本地开发完成,想部署到服务器上,选用了GIT来在服务器上获取上传的本地项目,结果clone后,访问网址后,YII就是个空白页,啥信息也没有,无语.. 刚开始以为是权限问题,后来给访问的目录加了777, ...

  7. var变量

    # Aduthor:CCIP-Ma name = "ma" name2 = name name = "ccip-ma" print("My name ...

  8. Java自学-集合框架 List接口

    ArrayList与List接口 步骤 1 : ArrayList和List ArrayList实现了接口List 常见的写法会把引用声明为接口List类型 注意:是java.util.List,而不 ...

  9. DesignPattern系列__07合成复用原则

    基本介绍 合成复用原则的核心,就是尽量去使用组合.聚合等方式,而不是使用继承. 核心思想 1.找出应用中可能需要变化之处,把它们独立出来,不要和那些不需要变化的代码混在一起. 2.针对接口编程,而不是 ...

  10. 实战讲解XXE漏洞的利用与防御策略

    现在许多不同的客户端技术都可以使用XMl向业务应用程序发送消息,为了使应用程序使用自定义的XML消息,应用程序必须先去解析XML文档,并且检查XML格式是否正确.当解析器允许XML外部实体解析时,就会 ...