背景:程序部署在客户机器上,不定期异常崩溃,且无日错误异常日志记录。
day1:初步排查是内存问题导致的,考虑使用分析工具记录分析。另外代码review仔细排查,怀疑有可能跟大量网络socket没有释放有关。
程序种使用到了httpclient,其中用的较多的get请求设置的超时时间如下:
RequestConfig requestConfig = RequestConfig.custom() .setConnectTimeout(180000).setConnectionRequestTimeout(180000) .setSocketTimeout(180000).build(); request.setConfig(requestConfig); CloseableHttpClient httpClient = getHttpClient();
三个超时时间分别代表,连接池获取可用连接超时,连接超时,读取数据超时
Httpclient源码如下:
RequestConfig requestConfig = RequestConfig.custom().setConnectionRequestTimeout(config.connReqTimeout) //从连接池中获取连接的超时时间 //与服务器连接超时时间:httpclient会创建一个异步线程用以创建socket连接,此处设置该socket的连接超时时间 .setConnectTimeout(config.connTimeout) .setSocketTimeout(config.socketTimeout) //socket读数据超时时间:从服务器获取响应数据的超时时间 .build(); httpClient = HttpClientBuilder.create() .setMaxConnTotal(config.maxConnTotal) //连接池中最大连接数 /** * 分配给同一个route(路由)最大的并发连接数。 * route:运行环境机器 到 目标机器的一条线路。 * 举例来说,我们使用HttpClient的实现来分别请求 www.baidu.com 的资源和 www.bing.com 的资源那么他就会产生两个route。 */ .setMaxConnPerRoute(config.maxConnPerRoute) .setDefaultRequestConfig(requestConfig) .build();
指的是连接目标url的连接超时时间,即客服端发送请求到与目标url建立起连接的最大时间。如果在该时间范围内还没有建立起连接,则就抛出connectionTimeOut异常。
如测试的时候,将url改为一个不存在的url:“http://test.com” ,
超时时间3000ms过后,系统报出异常:
org.apache.commons.httpclient.ConnectTimeoutException:The host did not accept the connection within timeout of 3000 ms
3.等待响应超时(读取数据超时)socketTimeout :
连接上一个url后,获取response的返回等待时间 ,即在与目标url建立连接后,等待放回response的最大时间,在规定时间内没有返回响应的话就抛出SocketTimeout。
测试的时候的连接url为我本地开启的一个url,http://localhost:8080/firstTest.htm?method=test,在我这个测试url里,当访问到这个链接时,线程sleep一段时间,来模拟返回response超时。
怀疑是以下代码设置超时时间过长导致的socket资源一直没有释放导致的。先调整超时时间观察一下。后续观察内存情况。