okhttp请求超时无效问题
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了okhttp请求超时无效问题相关的知识,希望对你有一定的参考价值。
参考技术A OKHttp请求超时无效问题Okhttp在网络请示出现错误时会重新发送请求,最终会不断执行
解决方式:retryOnConnectionFailure返回false
sClient = builder.retryOnConnectionFailure(false).build();
该问题 在3.4.2版本已处理
如何使用 OkHttp 设置连接超时
【中文标题】如何使用 OkHttp 设置连接超时【英文标题】:How to set connection timeout with OkHttp 【发布时间】:2014-11-15 05:21:25 【问题描述】:我正在使用 OkHttp 库开发应用程序,我的问题是我找不到如何设置连接超时和套接字超时。
OkHttpClient client = new OkHttpClient();
Request request = new Request.Builder().url(url).build();
Response response = client.newCall(request).execute();
【问题讨论】:
您选择的答案不再有效。也许您应该编辑问题以指定 okhttp 的版本。 【参考方案1】:从 OkHttp3 开始,您可以像这样通过Builder 进行此操作
client = new OkHttpClient.Builder()
.connectTimeout(10, TimeUnit.SECONDS)
.writeTimeout(10, TimeUnit.SECONDS)
.readTimeout(30, TimeUnit.SECONDS)
.build();
您也可以查看食谱here。
对于旧版本,您只需这样做
OkHttpClient client = new OkHttpClient();
client.setConnectTimeout(15, TimeUnit.SECONDS); // connect timeout
client.setReadTimeout(15, TimeUnit.SECONDS); // socket timeout
Request request = new Request.Builder().url(url).build();
Response response = client.newCall(request).execute();
请注意,setReadTimeout
中设置的值是 Socket
上 setSoTimeout
中使用的值,OkHttp
Connection
类内部。
未在OkHttpClient
上设置任何超时等效于在setConnectTimeout
或setReadTimeout
上设置0
的值,并且根本不会导致超时。描述可以在here找到。
正如@marceloquinta 在 cmets 中提到的,setWriteTimeout
也可以设置。
@ChristerNordvik 提到,从版本2.5.0
开始,读/写/连接超时值默认设置为 10 秒。这个可以看here。
【讨论】:
有没有办法为每个请求设置不同的超时时间(每个请求都有相同的静态客户端)? 您可以为每个请求设置特定的超时时间:这里有一些信息:github.com/square/okhttp/wiki/Recipes 不要忘记写超时:client.setWriteTimeout(10, TimeUnit.SECONDS); @MiguelLavigne 仅供参考,OkHttp 在 2.5.0 中更改为默认超时 10 秒。 过时的答案【参考方案2】:对于 okhttp3,这有点改变。
现在您使用构建器而不是设置器来设置时间,如下所示:
OkHttpClient client = new OkHttpClient.Builder()
.connectTimeout(10, TimeUnit.SECONDS)
.writeTimeout(10, TimeUnit.SECONDS)
.readTimeout(30, TimeUnit.SECONDS)
.build();
更多信息可以在他们的 wiki 中找到: https://github.com/square/okhttp/blob/b3dcb9b1871325248fba917458658628c44ce8a3/docs/recipes.md#timeouts-kt-java
【讨论】:
【参考方案3】:对于 Retrofit retrofit:2.0.0-beta4
,代码如下:
OkHttpClient client = new OkHttpClient.Builder()
.addInterceptor(logging)
.connectTimeout(30, TimeUnit.SECONDS)
.readTimeout(30, TimeUnit.SECONDS)
.writeTimeout(30, TimeUnit.SECONDS)
.build();
Retrofit retrofit = new Retrofit.Builder()
.baseUrl("http://api.yourapp.com/")
.addConverterFactory(GsonConverterFactory.create())
.client(client)
.build();
【讨论】:
【参考方案4】:对于Retrofit 2.0.0-beta1
或beta2
,代码如下:
OkHttpClient client = new OkHttpClient();
client.setConnectTimeout(30, TimeUnit.SECONDS);
client.setReadTimeout(30, TimeUnit.SECONDS);
client.setWriteTimeout(30, TimeUnit.SECONDS);
Retrofit retrofit = new Retrofit.Builder()
.baseUrl("http://api.yourapp.com/")
.addConverterFactory(GsonConverterFactory.create())
.client(client)
.build();
【讨论】:
【参考方案5】:现在变了。将.Builder()
替换为.newBuilder()
从 okhttp:3.9.0 开始,代码如下:
OkHttpClient okHttpClient = new OkHttpClient()
.newBuilder()
.connectTimeout(10,TimeUnit.SECONDS)
.writeTimeout(10,TimeUnit.SECONDS)
.readTimeout(30,TimeUnit.SECONDS)
.build();
【讨论】:
【参考方案6】:添加 gradle 文件并同步项目:
compile 'com.squareup.okhttp3:okhttp:3.2.0'
compile 'com.google.code.gson:gson:2.6.2'
在Java类中添加:
import okhttp3.OkHttpClient;
import okhttp3.OkHttpClient.Builder;
Builder b = new Builder();
b.readTimeout(200, TimeUnit.MILLISECONDS);
b.writeTimeout(600, TimeUnit.MILLISECONDS);
// set other properties
OkHttpClient client = b.build();
【讨论】:
【参考方案7】:OkHttp 版本:3.11.0
或更高版本
来自okhttp源码:
/**
* Sets the default connect timeout for new connections. A value of 0 means no timeout,
* otherwise values must be between 1 and @link Integer#MAX_VALUE when converted to
* milliseconds.
*
* <p>The connectTimeout is applied when connecting a TCP socket to the target host.
* The default value is 10 seconds.
*/
public Builder connectTimeout(long timeout, TimeUnit unit)
connectTimeout = checkDuration("timeout", timeout, unit);
return this;
unit
可以是以下任意值
TimeUnit.NANOSECONDS
TimeUnit.MICROSECONDS
TimeUnit.MILLISECONDS
TimeUnit.SECONDS
TimeUnit.MINUTES
TimeUnit.HOURS
TimeUnit.DAYS
示例代码
OkHttpClient client = new OkHttpClient.Builder()
.connectTimeout(5000, TimeUnit.MILLISECONDS)/*timeout: 5 seconds*/
.build();
String url = "https://www.google.com";
Request request = new Request.Builder()
.url(url)
.build();
try
Response response = client.newCall(request).execute();
catch (IOException e)
e.printStackTrace();
更新
我已经从3.12.0
版本向 OkHttp 添加了新的 API,你可以像这样设置超时:
OkHttpClient client = new OkHttpClient.Builder()
.connectTimeout(Duration.ofSeconds(5)) // timeout: 5 seconds
.build();
注意:这需要 API 26+
,因此如果您支持旧版本的 Android,请继续使用 (5, TimeUnit.SECONDS)
。
【讨论】:
【参考方案8】:像这样:
//New Request
HttpLoggingInterceptor logging = new HttpLoggingInterceptor();
logging.setLevel(HttpLoggingInterceptor.Level.BASIC);
final OkHttpClient client = new OkHttpClient.Builder()
.addInterceptor(logging)
.connectTimeout(30, TimeUnit.SECONDS)
.readTimeout(30, TimeUnit.SECONDS)
.writeTimeout(30, TimeUnit.SECONDS)
.build();
【讨论】:
【参考方案9】:这对我有用:
OkHttpClient client = new OkHttpClient.Builder()
.connectTimeout(10, TimeUnit.SECONDS)
.readTimeout(10, TimeUnit.SECONDS)
.writeTimeout(10, TimeUnit.SECONDS)
.retryOnConnectionFailure(false) <-- not necessary but useful!
.build();
来源:https://github.com/square/okhttp/issues/3553
【讨论】:
【参考方案10】:如果您想自定义配置,请先使用以下方法创建 OKhttpclient,然后在其上添加构建器。
private final OkHttpClient client = new OkHttpClient();
// Copy to customize OkHttp for this request.
OkHttpClient client1 = client.newBuilder()
.readTimeout(500, TimeUnit.MILLISECONDS)
.build();
try (Response response = client1.newCall(request).execute())
System.out.println("Response 1 succeeded: " + response);
catch (IOException e)
System.out.println("Response 1 failed: " + e);
【讨论】:
【参考方案11】:您可以设置调用超时,以涵盖从解析 DNS、连接、写入请求正文、服务器处理和读取响应正文的整个周期。
val client = OkHttpClient().newBuilder().callTimeout(CALL_TIMEOUT_IN_MINUTES, TimeUnit.MINUTES).build()
【讨论】:
以上是关于okhttp请求超时无效问题的主要内容,如果未能解决你的问题,请参考以下文章