作者简介:ASCE1885, 《Android 高级进阶》作者。

本文由于潜在的商业目的,未经授权不开放全文转载许可,谢谢!

本文分析的源码版本已经 fork 到我的 Github。

超时机制在现实世界中广泛存在,例如为了保证系统的正常有序运行,高铁等交通工具每个班次是有明确的出发时间和到达时间的,当你超过出发时间还没有上车时,那么不好意思,高铁系统的超时机制将发挥作用,正常情况下你将错过这趟列车。在计算机的世界中,同样如此,例如服务器对外提供的接口一般都设置了超时时间,但超过指定时间接口的请求还没有处理完成时,服务器将会主动断开连接,避免存在大量这种非正常的连接时拖垮服务器。客户端在请求服务器的接口时,同样会设置读取超时时间,当服务器由于某些原因迟迟未返回结果时,客户端为了保证用户体验会断开这次连接,并提示用户某些功能暂时不可用,稍后重试。

在本系列第一篇文章中我们提到 okio 中的输入流 Source 和输出流 Sink 的一大特点是引入了超时机制,和上面同样的道理,此处的超时机制也是为了保证系统正常有序的运行,本文就来聊聊它的原理和具体实现。okio 中超时机制主要有三种:

  • 同步超时 Timeout

  • 异步超时 AsyncTimeout

  • 基于装饰者模式的超时 ForwardingTimeout

其中 ForwardingTimeout 是使用装饰者模式对 Timeout 的封装,跟本系列第二篇文章中介绍的 ForwardingSource 类是同一个道理,不再赘述。

同步超时

同步超时用来控制某个任务执行的最大时长,当执行超过指定的时间时,任务将被中断,例如当从输入流 Source 中读取数据超时后,输入流将被关闭,读取操作只能稍后重试;当将数据写入输出流 Sink 超时时,输出流也将被关闭并等待稍后重试。同步超时 Timeout 类中用来判断是否超时存在两个策略:

  • 根据任务处理的超时时间 timeoutNanos 判断

  • 根据任务的截止时间点 deadlineNanoTime 判断

需要注意一点,deadlineNanoTime 表示的是某个具体的时间点,而 timeoutNanos 表示的是一段时间间隔。相关变量定义如下代码所示:

private boolean hasDeadline; // 是否设置了截止时间点
private long deadlineNanoTime; // 截止时间点(单位纳秒)
private long timeoutNanos; // 任务处理的超时时间间隔(单位纳秒)

关于这几个变量的设置和获取方法比较简单,我们略过不谈,直接来看下判断是否超时的方法 throwIfReached ,操作很明了,主要有两个判断条件:

  • 判断当前线程是否已经被中断

  • 判断当前时间点是否大于设定的截止时间点 deadlineNanoTime

如果满足条件,说明超时了,抛出 InterruptedIOException 表示超时时间到,代码如下所示:

public void throwIfReached() throws IOException {  if (Thread.interrupted()) { throw new InterruptedIOException("thread interrupted");   }   if (hasDeadline && deadlineNanoTime - System.nanoTime() <= 0) { throw new InterruptedIOException("deadline reached"); }
}

可以看到这个方法并没有根据 timeoutNanos 来判断是否超时,因此,当你的 Timeout 实例只设置了任务处理的超时时间 timeoutNanos 时,调用 throwIfReached 方法其实是不会发生超时操作的,这点 okio 的设计是存在问题的。

当然 timeoutNanos 不是说没有用到,它在接下来介绍的 waitUntilNotified 方法和异步超时中会使用到。waitUntilNotified 方法用来等待某个指定的 monitor 对象,直到这个对象被 notify 或者超时时间到,也属于同步超时的一种。

public final void waitUntilNotified(Object monitor) throws InterruptedIOException {   try {   boolean hasDeadline = hasDeadline();   long timeoutNanos = timeoutNanos();    // 当没有设置超时时间,将无限等待直到对象被notify    if (!hasDeadline && timeoutNanos == 0L) { monitor.wait(); // There is no timeout: wait forever.   return; }   // 根据timeoutNanos和deadlineNanoTime计算出较短的超时时间waitNanos   long waitNanos; long start = System.nanoTime();    if (hasDeadline && timeoutNanos != 0) {    long deadlineNanos = deadlineNanoTime() - start;   waitNanos = Math.min(timeoutNanos, deadlineNanos); } else if (hasDeadline) {   waitNanos = deadlineNanoTime() - start;    } else {    waitNanos = timeoutNanos;  }   // 调用wait方法尝试等待超时时间waitNanos,当然如果monitor对象被外界notify,  // 那么自然就不会傻傻的等到waitNanos超时了 long elapsedNanos = 0L;    if (waitNanos > 0L) {    long waitMillis = waitNanos / 1000000L;    // wait方法第一个参数表示等待的毫秒数,第二个参数表示等待的纳秒数,最终等待时间是两者之和  monitor.wait(waitMillis, (int) (waitNanos - waitMillis * 1000000L));    elapsedNanos = System.nanoTime() - start;  }   // 走到这里,说明wait等待超时时间到,或者monitor被外界notify了 if (elapsedNanos >= waitNanos) {    // 如果时超时时间到就抛出InterruptedIOException异常  throw new InterruptedIOException("timeout");  }   } catch (InterruptedException e) {  throw new InterruptedIOException("interrupted");  }
}

异步超时

异步超时 AsyncTimeout 继承自同步超时 Timeout,因此具有 Timeout 类的所有功能。此外,AsyncTimeout 使用一个后台线程 WatchDog 来实现超时的触发,相比同步超时而言,异步超时一般用来给原生不支持超时机制的类增加超时功能,例如 socket 在读写数据时本身是不支持超时的,因此,使用 okio 从 socket 中读写数据时,如果想要支持超时,可以通过 AsyncTimeout 给它增加这个功能,下面我们以从 socket 中读取数据为例进行介绍。AsyncTimeout 的子类通过重写 timedOut 方法来实现超时发生时的自定义处理逻辑,具体到 socket 这个例子,超时发生时我们自然是希望关闭 socket 连接,封装了 socket 的 AsyncTimeout 如下所示:

private static AsyncTimeout timeout(final Socket socket) { return new AsyncTimeout() { ... @Override protected void timedOut() {  try {   socket.close(); } catch (Exception e) { logger.log(Level.WARNING, "Failed to close timed out socket " + socket, e);  } catch (AssertionError e) {    if (isAndroidGetsocknameError(e)) { logger.log(Level.WARNING, "Failed to close timed out socket " + socket, e);  } else {    throw e;    }   }   }   };
}

了解了 AsyncTimeout 子类定义方式后,我们来进一步看看 AsyncTimeout 的具体实现。除了上面介绍的需要子类实现的 timeOut 方法,AsyncTimeout 另外的两个核心方法分别是:

  • enter:使用者在执行可能发生超时的任务前,需要调用这个方法

  • exit:使用者在执行可能发生超时的任务后,需要调用这个方法

在具体介绍这两个方法之前,我们先来了解下后台线程 WatchDog 和它管理的 AsyncTimeout 链表,结构如下图所示:

可以看到,所有的 AsyncTimeout 在底层会互相联结成为一个链表,链表的顺序按照超时时间由短到长链接,而 WatchDog 用来管理这张链表。链表定义相关代码如下:

public class AsyncTimeout extends Timeout {    // 链表的头节点,指向链表中第一个元素 static @Nullable AsyncTimeout head;    // 当前 AsyncTimeout 指向的下一个链表元素   private @Nullable AsyncTimeout next;
}

WatchDog 是一个守护线程,它的作用是当 AsyncTimeout 链表不为空时,轮询 AsyncTimeout 链表并触发最先超时的 AsyncTimeout 元素中的 timeOut 方法,代码如下所示:

... 更多内容请点击阅读原文继续阅读。

11 | Android 高级进阶(源码剖析篇) Square 高效易用的 IO 框架 okio(四)相关推荐

  1. CTS(11)---android自动化测试CTS源码分析之一

    android自动化测试CTS源码分析之一 1, 概述 CTS(Compatibility Test Suite)全名兼容性测试,主要目的就是让Android设备开发商能够开发出兼容性更好的andro ...

  2. android状态机是线程么,[Android] 状态机 StateMachine 源码剖析

    1. 案例 案例:我们常见的汽车,我们可以使用它行驶,也可以将它停止在路边.当它在行驶的过程中,需要不断的检测油量,一旦油量不足的时候,就将陷入停止状态.而停止在路边的汽车,需要点火启动,此时将检测车 ...

  3. c++11 shared_ptr 与 make_shared源码剖析

    写在最前... 请支持原创~~ 0. 前言 所谓智能指针,可以从字面上理解为"智能"的指针.具体来讲,智能指针和普通指针的用法是相似的,不同之处在于,智能指针可以在适当时机自动释放 ...

  4. Android Hawk的源码解析,一款基于SharedPreferences的存储框架

    转载请标注:http://blog.csdn.net/friendlychen/article/details/76218033 一.概念 SharedPreferences的使用大家应该非常熟悉啦. ...

  5. Android 人脸解锁源码剖析

    和你一起终身学习,这里是程序员Android 经典好文推荐,通过阅读本文,您将收获以下知识点: 一.人脸识别身份验证HIDL 二.人脸模块流程分析 三.人脸录入 四.人脸匹配 五.人脸解锁屏幕 一.人 ...

  6. 从Android 6.0源码的角度剖析View的绘制原理

    在从Android 6.0源码的角度剖析Activity的启动过程和从Android 6.0源码的角度剖析Window内部机制原理的文章中,我们分别详细地阐述了一个界面(Activity)从启动到显示 ...

  7. Mongoose源码剖析:外篇之web服务器

    引言 在深入Mongoose源码剖析之前,我们应该清楚web服务器是什么?它提供什么服务?怎样提供服务?使用什么协议?客户端如何唯一标识web服务器的资源?下面我们抛开Mongoose,来介绍一个we ...

  8. Vue 3源码剖析,看这篇就够了

    大家好,我是若川.源码的重要性相信不用再多说什么了吧,特别是用Vue 框架的,一般在面试的时候面试官多多少少都会考察源码层面的内容,比如: 如何理解虚拟Dom? Vue 3为什么这么快? Vue 3的 ...

  9. tomcat(11)org.apache.catalina.core.StandardWrapper源码剖析

    [0]README 0.0)本文部分文字描述转自 "how tomcat works",旨在学习 "tomcat(11)StandardWrapper源码剖析" ...

最新文章

  1. Yii框架 ajax案例
  2. Sql养成一个好习惯是一笔财富
  3. STM32通用定时器输出PWM控制舵机 —— 重装载值、比较值、当前值
  4. 批量获取域名解析地址socketthread
  5. 关于SQLSERVER的全文目录跟全文索引的区别
  6. MySQL sql_model问题研究
  7. feign扫描_微服务通信之feign的注册、发现过程
  8. PL/SQL Step By Step(三)
  9. 2、Linux多线程,线程的分离与结合
  10. IIS安装时,安装程序无法复制一个或多个文件。特定错误码是0x4b8
  11. 【枚举排列】生成1~n的排列生成可重集的排列
  12. 熊族部落---要邀请码
  13. 文件生成BASE64,base64转文件
  14. RSS阅读器Reeder简单使用攻略
  15. 158玩游戏平台最新上线
  16. 关于近日番茄花园洪磊被拘之事的一些个人看法
  17. 2018年最新old男孩python全栈第九期课程-大牛编程吧-Python编程精品区-大牛编程吧
  18. oracle怎么绑定vue,Oracle AutoVue 安装与配置教程,oracleautovue
  19. 小程序学习 - 02 微信小程序案例实践
  20. 小组取什么名字好_寓意兴旺的公司名字取什么名字好

热门文章

  1. 移动应用安全架构设计
  2. MATLAB学习笔记之matlab程序流程控制
  3. 函数恒成立问题与参数的取值范围_高考中恒成立问题巧解方法
  4. 机器学习算法(十三):word2vec
  5. Python是“真火”还是“虚火”?
  6. php $_SERVER的参数与说明
  7. servlet+ajax在线生成二维码
  8. 阻塞IO和非阻塞IO的区别 (BIONIO)
  9. pdf转换器的作用,这些你需要知道!
  10. thinkphp(tp)中paginate方法的学习