HttpClient java.net.SocketException: Connection reset

最近使用httpclient发送get请求时一直报错

总所周知http1.1(http1.0不是标准,依服务器而定)是支持长连接的,长连接能够保证服务器和客户端的socket能够高效利用,减少握手等额外的开销。httpClient在正常情况下会带上Connection: keep-alive表示我是支持长连接的,当完成一个请求后,视情况决定是否关闭连接。 
但是如果服务器端连接次数计数达到指定值时,则会在返回内容中添加Connection: close信息,表示该连接将被关闭。

这个时候该connection就失效了,客户端如果下次请求服务器的话,需要重新创建一个新的连接。但是有个问题,是否连接保持是由服务器端决定的,一旦连接超时导致connection close

小结:如果服务器端的过期时间小于客户端的过期时间,在临界值状态下就可能会发生如下报错: 
错误1:java.net.SocketException: Connection reset 
错误2:java.net.SocketException: Broken pipe 
错误3:org.apache.http.NoHttpResponseException: *** failed to respond 
会导致访问出问题,所以个人建议客户端的过期时间要小于服务器端的过期时间

 

对于java.net.SocketException: Connection reset 导致原因很多,网上得解决方案有好几种:

1.      使用TCP短链接无效。


 httppost.setProtocolVersion(HttpVersion.HTTP_1_0);
 httppost.addHeader(HTTP.CONN_DIRECTIVE, HTTP.CONN_CLOSE);

2.      链接时间改再大也无效
        int timeout = 60;//这个要弄长点
        RequestConfig defaultRequestConfig = RequestConfig.custom()
                .setSocketTimeout(timeout * 1000)
                .setConnectTimeout(timeout * 1000)
                .setConnectionRequestTimeout(timeout * 1000)
                .build();

3.设置keepalive

ConnectionKeepAliveStrategy kaStrategy = new DefaultConnectionKeepAliveStrategy() {
			@Override
			public long getKeepAliveDuration(HttpResponse response, HttpContext context) {
				long keepAlive = super.getKeepAliveDuration(response, context);
				if (keepAlive == -1) {
					//如果服务器没有设置keep-alive这个参数,我们就把它设置成1分钟
					keepAlive = 60000;
				}
				return keepAlive;
			}

		};

		CloseableHttpClient httpClient = HttpClients.custom().setKeepAliveStrategy(kaStrategy).build();

上面方法都试过了。并没解决我得问题。。。。。。。。

重新研究了下,当你请求得url会发生重定向时(请求返回得code = 301,302,307这样得),因为httpclient默认是会自动重定向得(get请求默认自动重定向。post不自动重定向),并且有重连机制,会导致前一个connection被重置。从而报错(大概是这样?不确定)特别是多次重定向。比如你的url请求时先重定向到A,再从A重定向到B,多次重定向会导致connection不断被重置或者被close。解决方案是关掉自动重定向。

这样可以关闭自动重定向,自己根据返回的response,从header中拿到Loaction,根据Location去再次请求重定向后的页面。