服务间基于HTTP通讯相对于grpc、dubbo之类的通讯效率要低得多,一方面是后者的传输数据结构紧凑,使用了序列化和压缩;另外一方面,后者使用了TCP链接池,而前者默认状况下每一次服务间的通讯会建立一个新的HTTP请求,会产生不小的性能消耗,对于须要额外非对称加密的HTTPS请求,性能消耗更加严重。java
默认状况下springboot的RestTemplate
使用的org.springframework.http.client.SimpleClientHttpRequestFactory
,即对每一次HTTP请求均新建一个新的TCP链接,请求结束后则关闭该链接。spring
RestTemplate restTemplate = new RestTemplate();
for(int i=0;i<100;i++) {
ResponseEntity<String> responseEntity = restTemplate.getForEntity("https://wakzz.cn", String.class);
System.out.println(responseEntity.getBody());
}
复制代码
经过抓包以下图,每次的HTTP请求都会断开上次的TCP链接,而后从新3次握手新建一个TCP链接。apache
HttpComponentsClientHttpRequestFactory
默认使用HTTP链接池PoolingHttpClientConnectionManager
,其默认参数maxTotal
为10即链接池最大HTTP链接数量为10,maxPerRoute
为2即链接池中相同目标IP和端口号的HTTP链接数量最多为2。c#
HttpComponentsClientHttpRequestFactory httpClientFactory = new HttpComponentsClientHttpRequestFactory();
httpClientFactory.setConnectTimeout(2000);
httpClientFactory.setReadTimeout(10000);
RestTemplate restTemplate = new RestTemplate(httpClientFactory);
for(int i=0;i<100;i++) {
ResponseEntity<String> responseEntity = restTemplate.getForEntity("https://wakzz.cn", String.class);
System.out.println(responseEntity.getBody());
}
复制代码
或者用户自定义PoolingHttpClientConnectionManager
的参数:springboot
PoolingHttpClientConnectionManager poolingConnectionManager = new PoolingHttpClientConnectionManager();
poolingConnectionManager.setMaxTotal(100);
poolingConnectionManager.setDefaultMaxPerRoute(10);
HttpClient httpClient = HttpClientBuilder.create()
.setConnectionManager(poolingConnectionManager)
.build();
HttpComponentsClientHttpRequestFactory httpClientFactory = new HttpComponentsClientHttpRequestFactory();
httpClientFactory.setConnectTimeout(2000);
httpClientFactory.setReadTimeout(10000);
httpClientFactory.setHttpClient(httpClient);
RestTemplate restTemplate = new RestTemplate(httpClientFactory);
for(int i=0;i<100;i++) {
ResponseEntity<String> responseEntity = restTemplate.getForEntity("https://wakzz.cn", String.class);
System.out.println(responseEntity.getBody());
}
复制代码
经过抓包以下图,每次的HTTP请求都复用的同一个TCP链接,无需在创建TCP链接上花费多余的性能消耗。数据结构
链接池链接状态:源码分析
HttpComponentsClientHttpRequestFactory
缺省值状况下在org.apache.http.impl.client.HttpClientBuilder#build
建立HTTP链接池:性能
final PoolingHttpClientConnectionManager poolingmgr = new PoolingHttpClientConnectionManager(
RegistryBuilder.<ConnectionSocketFactory>create()
.register("http", PlainConnectionSocketFactory.getSocketFactory())
.register("https", sslSocketFactoryCopy)
.build(),
null,
null,
dnsResolver,
connTimeToLive,
connTimeToLiveTimeUnit != null ? connTimeToLiveTimeUnit : TimeUnit.MILLISECONDS);
if (defaultSocketConfig != null) {
poolingmgr.setDefaultSocketConfig(defaultSocketConfig);
}
if (defaultConnectionConfig != null) {
poolingmgr.setDefaultConnectionConfig(defaultConnectionConfig);
}
if (systemProperties) {
String s = System.getProperty("http.keepAlive", "true");
if ("true".equalsIgnoreCase(s)) {
s = System.getProperty("http.maxConnections", "5");
final int max = Integer.parseInt(s);
poolingmgr.setDefaultMaxPerRoute(max);
poolingmgr.setMaxTotal(2 * max);
}
}
if (maxConnTotal > 0) {
poolingmgr.setMaxTotal(maxConnTotal);
}
if (maxConnPerRoute > 0) {
poolingmgr.setDefaultMaxPerRoute(maxConnPerRoute);
}
复制代码
org.apache.http.impl.execchain.MainClientExec#execute
从链接池获取HTTP链接:ui
final ConnectionRequest connRequest = connManager.requestConnection(route, userToken);
if (execAware != null) {
if (execAware.isAborted()) {
connRequest.cancel();
throw new RequestAbortedException("Request aborted");
} else {
execAware.setCancellable(connRequest);
}
}
final RequestConfig config = context.getRequestConfig();
final HttpClientConnection managedConn;
try {
final int timeout = config.getConnectionRequestTimeout();
managedConn = connRequest.get(timeout > 0 ? timeout : 0, TimeUnit.MILLISECONDS);
} catch(final InterruptedException interrupted) {
Thread.currentThread().interrupt();
throw new RequestAbortedException("Request aborted", interrupted);
} catch(final ExecutionException ex) {
Throwable cause = ex.getCause();
if (cause == null) {
cause = ex;
}
throw new RequestAbortedException("Request execution failed", cause);
}
复制代码
每次发起HTTP请求时经过路由route(目标IP和端口号)从链接池获取连接,默认timeout为0即从链接池无限等待获取链接不超时。加密