关于setConnectTimeout和setReadTimeout的问题

1、问题描述

因为用户登录记录需要记录登录地址,这个地址获取调用的是阿里的接口,登录的时候由于connect没有响应,造成一直卡在登录界面。

 

2、问题排查和定位

思考:网络差不应该报网络异常的错误或者直接抛timeout异常吗?所以马上去检查Http请求的地方.

 

备注:(在DownloadThread中使用HttpURLConnection进行网络请求时只是设置了connectTimeout,这样在建立连接的时候如果网络中断,那么超时以后就会报出错误,这里是没有问题的。会不会是还少设置了其他参数的原因呢?)

 

于是赶紧去查阅资料:

ConnectTimeout只有在网络正常的情况下才有效,而当网络不正常时,ReadTimeout才真正的起作用,即IdIOHandlerStack 里的 WaitFor 是受ReadTimeout限制的,因此,这2个属性应该结合实用。

 

connect timeout 是建立连接的超时时间;

read timeout,是传递数据的超时时间。

 

正确的方法应当是调用HttpURLConnection的时候同时设置这两个超时时间。

一看代码,果然漏了setReadTimeout。。。。。OMG 。。。

 

3、解决方案

同时设置这两个超时时间:

 
  1. URLConnection connection = url.openConnection();

  2. connection.setConnectTimeout(5000); // 5秒 连接主机的超时时间(单位:毫秒)

  3. connection.setReadTimeout(5000); // 5秒 从主机读取数据的超时时间(单位:毫秒)

参考:https://blog.csdn.net/jackson_wen/article/details/51923514

 

  • ConnectTimeout

    指的是建立连接所用的时间,适用于网络状况正常的情况下,两端连接所用的时间。
    java中,网络状况正常的情况下,例如使用HttpClient或者HttpURLConnetion连接时设置参数connectTimeout=5000即5秒,如果连接用时超过5秒就是抛出java.net.SocketException: connetct time out的异常
  • ReadTimeout

    指的是建立连接后从服务器读取到可用资源所用的时间。
    在这里我们可以这样理解ReadTimeout:正常情况下,当我们发出请求时可以收到请求的结果,也就是页面上展示的内容,但是当网络状况很差的时候,就会出现页面上无法展示出内容的情况。另外当我们使用爬虫或者其他全自动的程序时,无法判断当前的网络状况是否良好,此时就有了ReadTimeout的用武之地了,通过设置ReadTimeout参数,例:ReadTimeout=5000,超过5秒没有读取到内容时,就认为此次读取不到内容并抛出Java.net.SocketException: read time out的异常。
    根据上面关于ConnectTimeout和ReadTimeout的描述,在我们使用需要设置这两项参数的服务或程序时,应该对两项参数一起设置。

一般而言两项参数的数值可以设置成一样的,但根据笔者个人经验,可以把ReadTimeout设置的长一点,ConnectTimeout可以相对比较短,这是源于我们的网络状况一般较为稳定,连接时很少出现问题,但是读取时因为数据下载时的网络波动,出状况的可能性更大一些。

 

httpclient超时时间

说明

ConnectionRequestTimeout

httpclient使用连接池来管理连接,这个时间就是从连接池获取连接的超时时间,可以想象下数据库连接池

ConnectTimeout

连接建立时间,三次握手完成时间

SocketTimeout

数据传输过程中数据包之间间隔的最大时间

下面重点说下SocketTimeout,比如有如下图所示的http请求

这里写图片描述

虽然报文(“abc”)返回总共用了6秒,如果SocketTimeout设置成4秒,实际程序执行的时候是不会抛出java.net.SocketTimeoutException: Read timed out异常的。

因为SocketTimeout的值表示的是“a”、”b”、”c”这三个报文,每两个相邻的报文的间隔时间不能超过SocketTimeout

 

RestTemplate 是一个非常常用的 HTTP 客户端工具,但在使用过程中可能会遇到一些问题,以下是一些常见问题及解决方法: 1. 如何设置 RestTemplate 的请求头? 可以使用 HttpHeaders 类来设置请求头,例如: ``` RestTemplate restTemplate = new RestTemplate(); HttpHeaders headers = new HttpHeaders(); headers.set("Authorization", "Bearer token"); headers.set("Content-Type", "application/json"); HttpEntity<String> entity = new HttpEntity<>("body", headers); restTemplate.exchange(url, HttpMethod.POST, entity, String.class); ``` 2. 如何发送 GET 请求并接收 JSON 格式的响应? 可以使用 RestTemplate 的 getForObject 方法,例如: ``` RestTemplate restTemplate = new RestTemplate(); String result = restTemplate.getForObject(url, String.class); ``` 3. 如何发送 POST 请求并接收 JSON 格式的响应? 可以使用 RestTemplate 的 postForObject 方法,例如: ``` RestTemplate restTemplate = new RestTemplate(); HttpHeaders headers = new HttpHeaders(); headers.set("Content-Type", "application/json"); HttpEntity<String> entity = new HttpEntity<>("body", headers); String result = restTemplate.postForObject(url, entity, String.class); ``` 4. 如何处理返回的响应状态码? 可以通过 ResponseEntity 来获取响应状态码和响应体,例如: ``` RestTemplate restTemplate = new RestTemplate(); ResponseEntity<String> response = restTemplate.getForEntity(url, String.class); if (response.getStatusCode() == HttpStatus.OK) { String result = response.getBody(); } else { // 处理失败逻辑 } ``` 5. 如何处理请求超时? 可以通过设置 RestTemplate 的超时时间来处理请求超时,例如: ``` RestTemplate restTemplate = new RestTemplate(); SimpleClientHttpRequestFactory requestFactory = new SimpleClientHttpRequestFactory(); requestFactory.setConnectTimeout(5000); requestFactory.setReadTimeout(5000); restTemplate.setRequestFactory(requestFactory); ``` 以上是一些常见的 RestTemplate 使用问题及解决方法,希望对你有所帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值