美文网首页
ConnectionPoolTimeoutException

ConnectionPoolTimeoutException

作者: 程序员札记 | 来源:发表于2023-09-20 08:09 被阅读0次

    1 错误信息

    今天解决了一个HttpClient的异常。

    问题表现:

    tomcat后台日志发现大量异常

    org.apache.http.conn.ConnectionPoolTimeoutException: Timeout waiting for connection

    时间一长tomcat就无法继续处理其他请求,从假死变成真死了。
    linux运行:

    发现CLOSE_WAIT的数量始终急速增长,一直没降过。


    image.png

    问题分析:

    一开始我对我的HttpClient使用过程深信不疑,我不认为异常是来自这里。

    所以我开始从TCP的连接状态入手,猜测可能导致异常的原因。以前经常遇到TIME_WAIT数过大导致的服务器异常,很容易解决,修改下sysctl就ok了。但是这次是CLOSE_WAIT,是完全不同的概念了。

    简单来说CLOSE_WAIT数目过大是由于被动关闭连接处理不当导致的。

    我说一个场景,服务器A会去请求服务器B上面的apache获取文件资源,正常情况下,如果请求成功,那么在抓取完资源后服务器A会主动发出关闭连接的请求,这个时候就是主动关闭连接,连接状态我们可以看到是TIME_WAIT。如果一旦发生异常呢?假设请求的资源服务器B上并不存在,那么这个时候就会由服务器B发出关闭连接的请求,服务器A就是被动的关闭了连接,如果服务器A被动关闭连接之后自己并没有释放连接,那就会造成CLOSE_WAIT的状态了。

    看看调用它的代码是什么样的:

    很简单,就是个远程读取中文页面的方法。看上去没啥问题,是用于非200状态的异常处理:


    image.png

    代码本身没有问题,但是问题是放错了位置。如果这么写的话就没问题:


    image.png

    前面那种写法InputStream in 根本就不会被赋值,意味着一旦出现非200的连接,这个连接将永远僵死在连接池里头,太恐怖了。。。所以我们看到CLOST_WAIT数目为400,因为对一个路由的连接已经完全被僵死连接占满了。。。
    其实上面那段代码还有一个没处理好的地方,异常处理不够严谨,所以最后我把代码改成了这样:

    image.png

    显示调用HttpGet的abort,这样就会直接中止这次连接,我们在遇到异常的时候应该显示调用,因为谁能保证异常是在InputStream in赋值之后才抛出的呢。

    相关文章

      网友评论

          本文标题:ConnectionPoolTimeoutException

          本文链接:https://www.haomeiwen.com/subject/lsvovdtx.html