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 中设置的值是 SocketsetSoTimeout 中使用的值,OkHttp Connection 类内部。

未在OkHttpClient 上设置任何超时等效于在setConnectTimeoutsetReadTimeout 上设置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-beta1beta2,代码如下:

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请求超时无效问题的主要内容,如果未能解决你的问题,请参考以下文章

发布大型正文时,okHttp 总是超时

SpringBoot Java后端实现okhttp3超时设置

OkHttp全局拦截器设置token超时重新获取

android okhttp超时怎么办

http超时请求设置

http超时请求设置