HttpClient出现TCP连接异常关闭发送RST包

文章分析了使用HttpClient 4.5.2时出现的异常TCP关闭问题,即发送RST包而非正常四次挥手关闭连接。通过代码示例展示了如何正确地释放连接,避免不必要的资源浪费。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题现象

在与第三方系统通过http交互数据的过程中,抓包发现每次TCP连接都是异常关闭,报文如下: 输入图片说明

可以看到,由我方发起3次握手建立连接,然后发送http请求,对方响应数据,我方ACK后直接发RST包(图中蓝色阴影部分),将连接异常关闭,不仅没有复用连接,且不是通过4次挥手来关闭连接。

关于RST包参考:http://blog.youkuaiyun.com/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.youkuaiyun.com/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/

转载于:https://my.oschina.net/hebaodan/blog/1579967

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值