图文详解OkHttp的超时时间

作者:今天i你好吗 时间:2022-05-14 13:50:23 

目录
  • 前言

  • connectTimeout:

  • callTimeout:

  • pingInterval

  • writeTimeout

  • readTimeout

  • 总结

前言

虽然网上有很多关于okhttp超时时间的文章但大多都一笔带过并没有进行详细的讲解各自的作用,于是就看了下源码大致写一下其中的发现.

本文以 'com.squareup.okhttp3:okhttp:3.12.0'源码为参考

首先我们一共可以设置5个超时时间分别如下:

OkHttpClient client = new OkHttpClient.Builder()

.connectTimeout(30, TimeUnit.SECONDS)

.callTimeout(120, TimeUnit.SECONDS)

.pingInterval(5, TimeUnit.SECONDS)

.readTimeout(60, TimeUnit.SECONDS)

.writeTimeout(60, TimeUnit.SECONDS)

.build();

其中callTimeout,readTimeout,writeTimeout和okio的AsyncTimeout有着密不可分的关系,其内部维护了一个Watchdog,单独开一个线程死循环判断是否超时

图文详解OkHttp的超时时间

connectTimeout:

指的是建立连接所用的时间,适用于网络状况正常的情况下,两端连接所用的时间。

通过跟源码发现这个值用在了 socket.connect(address, connectTimeout);

callTimeout:

这个值从调用call.execute();和enqueue();这两个方法开始计时,时间到后网络还未请求完成将调用cancel();方法
在RealCall类中可以看到在构造方法中创建timeout匿名内部类

图文详解OkHttp的超时时间

在execute方法中开始计时

图文详解OkHttp的超时时间

在timeoutExit方法中结束计时

图文详解OkHttp的超时时间

pingInterval

通过跟源码我们可以看到,这个值只有http2和webSocket中有使用

图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

如果设置了这个值会定时的向服务器发送一个消息来保持长连接

图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

所以在写websocket时是完全可以只用设置这个值来保持长连接的.

客户端在发送ping消息时服务端会相应的返回pong消息来进行回应.同时okhttp也实现了pong,服务端在发起ping的时候客户端会通过pong来进行回应,即:在进行长连接时,客户端不需要进行只需要服务端进行定时ping也是可以保持长连接的.

接下来就开始讲和我们密切相关的readTimeout和writeTimeout了,当然也是最复杂的.其中最重要的还是readTimeout,我们先看writeTimeout

writeTimeout

这个值大致有3个地方用到

图文详解OkHttp的超时时间

其中第二处和第三处的用用法是一致的,最后都是调用了


sink.timeout().timeout(writeTimeout, MILLISECONDS);

这写到底是什么意思呢?

这个就不得不说okio了,okhttp中几乎所有的流的操作都是由okio完成的,在okio.AsyncTimeout中对Sink(类似于OutputStream)和Source(类似于InputStream)进行了一层封装

图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

/**

Don't write more than 64 KiB of data at a time, give or take a segment. Otherwise slow
connections may suffer timeouts even when they're making (slow) progress. Without this, writing
a single 1 MiB buffer may never succeed on a sufficiently slow connection.
*/
private static final int TIMEOUT_WRITE_SIZE = 64 * 1024;

这其中的逻辑还是相当复杂的,大致意思就是所有的sink都被封装了一个超时机制,需要在我们设置的时间内写出TIMEOUT_WRITE_SIZE(64k)的数据,如果无法完成即为超时,所以,我们在上次文件时明明只设置了几十秒的超时时间却不会超时.

在http2中就没有再使用okio的超时机制了,当然超时计时器还是用的AsyncTimeout.的Watchdog

图文详解OkHttp的超时时间
图文详解OkHttp的超时时间

图文详解OkHttp的超时时间
图文详解OkHttp的超时时间

图文详解OkHttp的超时时间
图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

可以看到,在http2中采用的是线程等待的策略

readTimeout

readTimeout和writeTimeout几乎完全一样,只是操作相反,而且header的读取和body的读取是分开进行的,由于header数据量较小就不用讨论了.

okio中每次读取不大于8k.


final class Segment {
/** The size of all segments in bytes. */
static final int SIZE = 8192;

图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

http2中每次读取不大于8k.

图文详解OkHttp的超时时间

图文详解OkHttp的超时时间

然后还漏了一点:

图文详解OkHttp的超时时间


socket.setSoTimeout(chain.readTimeoutMillis());

这行代码什么意思呢?

setSotimeout(10000)是表示如果对方连接状态10秒没有收到数据的话强制断开客户端。
如果想要长连接的话,可以使用心跳包来通知服务器,也就是我没有发给你数据,但是我告诉你我还活着.

最后,如果超时时间设置的如果是0,那么代表超时时长为无限.

附上okhttp的默认超时时间

图文详解OkHttp的超时时间

来源:https://www.jianshu.com/p/df7b31b4a624

标签:okhttp,超时,时间
0
投稿

猜你喜欢

  • Java内存溢出实现原因及解决方案

    2022-10-19 09:07:42
  • C#算法设计与分析详解

    2023-03-03 10:15:06
  • maven报错:Failed to execute goal on project问题及解决

    2021-09-30 09:58:36
  • 如何解决Java多线程死锁问题

    2022-08-11 15:51:02
  • 使用C# CefSharp Python采集某网站简历并且自动发送邀请短信的方法

    2023-11-17 14:09:34
  • C#泛型方法在lua中表示的一种设计详解

    2022-08-24 20:03:12
  • Android ImgView属性图文详解

    2023-02-21 11:14:18
  • android.enableD8.desugaring = false引发问题解决

    2023-11-03 16:06:15
  • Android文本输入框(EditText)输入密码时显示与隐藏

    2022-04-24 06:23:47
  • C#实现JSON字符串序列化与反序列化的方法

    2023-12-01 12:40:23
  • Android中使用Kotlin实现一个简单的登录界面

    2023-01-29 13:51:37
  • C#语言中字符类char的使用方法(总结)

    2022-09-04 17:25:43
  • 遍历Hashtable 的几种方法

    2023-07-18 13:49:11
  • IDEA如何自动生成serialVersionUID的设置

    2023-08-09 04:00:33
  • Java8中的LocalDateTime你会使用了吗

    2023-11-15 07:02:07
  • AndroidStudio kotlin配置详细介绍

    2022-02-18 14:09:21
  • android输入框与文本框加滚动条scrollview示例

    2023-07-29 09:22:42
  • 解决idea web 配置相对路径问题

    2022-01-12 06:48:40
  • Java并发编程示例(一):线程的创建和执行

    2022-01-24 16:43:49
  • spring boot的拦截器简单使用示例代码

    2021-09-29 04:07:07
  • asp之家 软件编程 m.aspxhome.com