问题现象
在与第三方系统通过http交互数据的过程中,抓包发现每次TCP连接都是异常关闭,报文如下:
可以看到,由我方发起3次握手建立连接,然后发送http请求,对方响应数据,我方ACK后直接发RST包(图中蓝色阴影部分),将连接异常关闭,不仅没有复用连接,且不是通过4次挥手来关闭连接。
关于RST包参考:http://blog.csdn.net/erlib/ar...
问题原因
HttpClient 4.5.2 释放连接API使用方式不正确
问题代码如下(已简化):
private boolean isCheckSuccess() {
CloseableHttpResponse response = null;
try {
// 通过HttpClientBuilder创建CloseableHttpClient
httpClient = HttpClientFactory.createHttpClient();
request = initCheckRequest();
response = httpClient.execute(request);
if(response.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {
return true;
}
} finally {
// 释放连接
if(request != null) {
request.releaseConnection();
}
}
return false;
}
//构造请求参数
private static HttpPost initCheckRequest() throws IOException {
HttpPost request = new HttpPost(callBackUrl);
request.setHeader("contentType", ContentTypeEnum.JSON.getDesc());
List<NameValuePair> params = new ArrayList<>();
params.add(new BasicNameValuePair("developer_id", "111"));
params.add(new BasicNameValuePair("sign", "sign"));
HttpEntity postParams = new UrlEncodedFormEntity(params);
request.setEntity(postParams);
return request;
}
根据释放连接处代码和抓包分析来看,request.releaseConnection();
每次是发TCP RST包来关闭连接
解决方案
既然是释放连接不对,则修改代码为如下:
finally {
if(response != null) {
try {
EntityUtils.consume(response.getEntity());
response.close();
} catch (IOException e) {
logger.error("close http error", e);
}
}
}
正确方式为:
关闭内容流后再关闭响应本身。
如果不关闭IO流,直接response.close();
则结果是连接不能复用,直接通过4次挥手断开TCP连接。
连接池释放连接的时候,并不会直接对TCP连接的状态有任何改变,只是维护了两个Set,leased和avaliabled,leased代表被占用的连接集合,avaliabled代表可用的连接的集合,释放连接的时候仅仅是将连接从leased中remove掉了,并把连接放到avaliabled集合中
本人blog:https://my.oschina.net/hebaod...
参考
TCP RST相关
http://www.cnblogs.com/JohnAB...
http://blog.csdn.net/erlib/ar...
http://lovestblog.cn/blog/201...
https://my.oschina.net/costax...
HttpClient相关
https://www.cnblogs.com/likai...
http://liangbizhi.github.io/h...
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。