HttpClient出现TCP连接异常关闭发送RST包
2017-11-26 00:00
585 查看
摘要: 在与第三方系统通过http交互数据的过程中,抓包发现每次TCP连接都是异常关闭,报文如下:
可以看到,由我方发起3次握手建立连接,然后发送http请求,对方响应数据,我方ACK后直接发RST包(图中蓝色阴影部分),将连接异常关闭,不仅没有复用连接,且不是通过4次挥手来关闭连接。
关于RST包参考:http://blog.csdn.net/erlib/article/details/50132307
问题代码如下(已简化):
根据释放连接处代码和抓包分析来看,
正确方式为:
关闭内容流后再关闭响应本身。
如果不关闭IO流,直接
连接池释放连接的时候,并不会直接对TCP连接的状态有任何改变,只是维护了两个Set,leased和avaliabled,leased代表被占用的连接集合,avaliabled代表可用的连接的集合,释放连接的时候仅仅是将连接从leased中remove掉了,并把连接放到avaliabled集合中
http://www.cnblogs.com/JohnABC/p/6323046.html
http://blog.csdn.net/erlib/article/details/50132307
http://lovestblog.cn/blog/2014/05/20/tcp-broken-pipe/
https://my.oschina.net/costaxu/blog/127394
HttpClient相关
https://www.cnblogs.com/likaitai/p/5431246.html
http://liangbizhi.github.io/httpclient-4-3-x-chapter-1/
问题现象
在与第三方系统通过http交互数据的过程中,抓包发现每次TCP连接都是异常关闭,报文如下:可以看到,由我方发起3次握手建立连接,然后发送http请求,对方响应数据,我方ACK后直接发RST包(图中蓝色阴影部分),将连接异常关闭,不仅没有复用连接,且不是通过4次挥手来关闭连接。
关于RST包参考:http://blog.csdn.net/erlib/article/details/50132307
问题原因
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集合中
参考
TCP RST相关http://www.cnblogs.com/JohnABC/p/6323046.html
http://blog.csdn.net/erlib/article/details/50132307
http://lovestblog.cn/blog/2014/05/20/tcp-broken-pipe/
https://my.oschina.net/costaxu/blog/127394
HttpClient相关
https://www.cnblogs.com/likaitai/p/5431246.html
http://liangbizhi.github.io/httpclient-4-3-x-chapter-1/
相关文章推荐
- HttpClient 异常出现不关闭引发的血案
- HttpClient 异常出现不关闭引发的血案
- HTTP的RST包与WinHttp延迟关闭TCP连接
- Golang http.DefaultClient 连接服务端压测时出现大量TIME_WAIT异常
- HttpClient——连接关闭
- tcp socket客户端发送请求连接http服务
- 使用HttpWebRequest进行请求时发生错误:基础连接已关闭,发送时发生错误处理
- 几种TCP连接中出现RST的情况
- C# Socket TcpClient 无法从传输连接中读取数据: 远程主机强迫关闭了一个现有的连接。。
- 几种TCP连接中出现RST的情况
- C#利用HttpWebRequest后台请求,出现:基础连接已经关闭: 接收时发生错误;【解决方法之一】
- 几种TCP连接中出现RST的情况
- 在向服务器发送请求时发生传输级错误。 (provider: TCP 提供程序, error: 0 - 远程主机强迫关闭了一个现有的连接。)
- System.Data.SqlClient.SqlException: 在向服务器发送请求时发生传输级错误。 (provider: TCP 提供程序, error: 0 - 远程主机强迫关闭了一个现有的连接。) .
- TCP连接的状态与关闭方式,及其对Server与Client的影响
- TCP/IP详解--TCP中异常关闭连接的意义 异常关闭的情况
- 原 几种TCP连接中出现RST的情况
- C#操作串口,带异常处理,解决端口占用时打开串口出现异常错误,串口关闭时发送数据报错
- C#操作串口,带异常处理,解决端口占用时打开串口出现异常错误,串口关闭时发送数据报错
- TCP/IP详解--TCP中异常关闭连接的意义 异常关闭的情况