Kotlin 中的suspend 关键字

简介: Kotlin 中的suspend 关键字

1,初认suspend


suspend 用于暂停执行当前协程,并保存所有局部变量,被标记为 suspend 的函数只能运行在协程或者其他 suspend 函数。

首先我们看一下在retrofit 不是使用suspend关键字会造成什么错误?



IllegalArgumentException: Unable to create call adapter for 
com.qxf.sample.network.BaseResponse

没有添加suspend关键字的时候回调数据不能创建返回的数据类型,类型错误了


添加上了suspend关键字,运行时会被编译成一个 Continuation

    @SinceKotlin("1.3")
    public interface Continuation<in T> {
        /**
         * Context of the coroutine that corresponds to this continuation.
         */
        public val context: CoroutineContext
        /**
         * 恢复协程的调用,将成功或者失败的结果回调出去
         */
        public fun resumeWith(result: Result<T>)
    }


    也可以认为是回调,这样比较直观一些;


    2,使用 suspend 函数无须关心线程切换


    用这个函数不会阻塞当前调用的线程。这对 UI 编程是非常有用的,因为 UI 的主线程需要不断相应各种图形绘制、用户操作的请求,如果主线程上有耗时操作会让其他请求无法及时响应,造成 UI 卡顿

      lifecycleScope.launch { 
      val posts = retrofit.get<PostService>().fetchPosts(); 
      // 由于在主线程,可以拿着 posts 更新 UI
      }

      这相比 callback 和 RxJava 的 API 是要好很多的。这些异步的 API 最终都得依靠回调,但回调回来在哪个线程需要调用方自己搞清楚,得看这些函数里面是怎么实现的。而有了 suspend 不阻塞当前线程的约定,调用方其实无须关心这个函数内部是在哪个线程执行的。


      lifecycleScope.launch(Dispatchers.Main) { 
      foo()  
      }

      如上面这个代码块,指定这个协程块调度到主线程执行,里面调用了一个不知道哪里来的 suspend foo 方法。这个方法内部可能是耗时的 CPU 计算,可能是耗时的 IO 请求,但是我在写这个协程块的时候,其实并不需要关心这里面到底是怎么回事,运行在哪个线程。类似地,在阅读这段协程块的时候,我们可以清楚地知道眼前的这段代码会在主线程执行,suspend foo 里面的代码是一个潜在的耗时操作,具体在哪个线程执行是这个函数的实现细节,对于当前代码的逻辑是「透明」的。

      但前提是这个 suspend 函数实现正确,真正做到了不阻塞当前线程。单纯地给函数加上 suspend 关键字并不会神奇地让函数变成非阻塞的,比如假设 suspend foo 里面的实现是这样的:


      suspend fun foo() = BigInteger.probablePrime(1024, Random())

      这个 foo 函数的实现没有遵守 suspend 的语义,是错误的。正确的做法应该修改这个 foo 函数:


        suspend fun findBigPrime(): BigInteger =  withContext(Dispatchers.Default) {  
          BigInteger.probablePrime(4096, Random())  
          }

        借助 withContext 我们把耗时操作从当前主线程挪到了一个默认的后台线程池,即使是用了协程,最终还是会「阻塞」某个线程,「所有的代码本质上都是阻塞式的」,这种理解可以帮助我们认识到 Android / JVM 上最终需要线程作为执行协程的载体,但忽略了阻塞和非阻塞 IO 之分,CPU 执行线程,而上面 BigInteger.probablePrime 是一个耗时的 CPU 计算,只能等待 CPU 把结果算出来,但 IO 造成的等待并不一定要阻塞 CPU,阻塞和非阻塞 IO 是有实际区别的。比如 Retrofit 虽然支持 suspend 函数(实际上也就是包装一下基于回调的 API enqueue),但是底层依赖的 OkHttp 用的是阻塞的方法,最终执行请求还是调度到线程池里面去.

        把协程和 suspend 单纯看成线程切换工具有很大的局限性。由于 suspend 就是回调,也提供了包装回调 API 的方法,基于回调的 API 都可以用 suspend 函数进行封装改造.

        相关文章
        |
        5月前
        |
        设计模式 Android开发 Kotlin
        Android经典实战之Kotlin委托模式和by关键字
        本文介绍了Kotlin中`by`关键字在类及属性委托中的运用,通过实例展示了如何利用类委托简化接口实现,以及如何借助标准与自定义属性委托管理属性的读写操作。通过`by`关键字的支持,Kotlin使得委托模式的实现更为直观且高效。
        111 4
        |
        6月前
        |
        SQL 安全 Java
        Android经典面试题之Kotlin中object关键字实现的是什么类型的单例模式?原理是什么?怎么实现双重检验锁单例模式?
        Kotlin 单例模式概览 在 Kotlin 中,`object` 关键字轻松实现单例,提供线程安全的“饿汉式”单例。例如: 要延迟初始化,可使用 `companion object` 和 `lazy` 委托: 对于参数化的线程安全单例,结合 `@Volatile` 和 `synchronized`
        78 6
        |
        API Kotlin
        Kotlin中扩展函数、infix关键字、apply函数和DSL的详解
        Kotlin中扩展函数、infix关键字、apply函数和DSL的详解
        151 0
        |
        Kotlin
        Kotlin中接口、抽象类、泛型、out(协变)、in(逆变)、reified关键字的详解
        Kotlin中接口、抽象类、泛型、out(协变)、in(逆变)、reified关键字的详解
        127 0
        |
        Kotlin
        Kotlin中继承、类型转换、Any超类、object关键字详解
        Kotlin中继承、类型转换、Any超类、object关键字详解
        190 0
        |
        Kotlin
        Kotlin 中定义类、field关键字,主构造函数和次构造函数详解
        Kotlin 中定义类、field关键字,主构造函数和次构造函数详解
        179 0
        |
        存储 Java 编译器
        Kotlin 学习笔记(四)—— 作用域函数、inline 关键字、反引号等 Kotlin 基本用法(下)
        Kotlin 学习笔记(四)—— 作用域函数、inline 关键字、反引号等 Kotlin 基本用法(下)
        65 0
        |
        Java Android开发 开发者
        Kotlin 学习笔记(四)—— 作用域函数、inline 关键字、反引号等 Kotlin 基本用法(上)
        Kotlin 学习笔记(四)—— 作用域函数、inline 关键字、反引号等 Kotlin 基本用法(上)
        114 0
        |
        Java 程序员 编译器
        01. Kotlin 标识符、关键字和注释
        01. Kotlin 标识符、关键字和注释
        151 0
        |
        Kotlin
        【Kotlin】Kotlin 委托 ( 使用 by 关键字进行接口委托 )
        【Kotlin】Kotlin 委托 ( 使用 by 关键字进行接口委托 )
        252 0