kotlin汇总7-Coroutines

在kotlin1.1,Coroutines还处于实验阶段。
有些API启动耗时操作比如:网络IO,文件流IO,CPU/GPU要求高的工作,它们会导致调用阻塞直到操作完成。Coroutines通过一种更便宜,更容易控制的suspension(挂起)操作,来避免线程阻塞。

Coroutines把复杂的运算移到了libraries,因而简化了异步编程。在Coroutine中,程序逻辑可以顺序表达(同步),底层的libraries可以帮我们完成异步操作,我们只需要在异步回调中编写我们的代码就完成了,可以在不同的线程订阅相关的事情,执行计划,代码就像顺序执行一样简单。

像其他语言中的大多数异步机制都可以通过coroutines实现。包括C#和ECMAScript的async/await,Go的channels和select, C#和Python的generator/yeild.

阻塞VS挂起
基本上,coroutines可以挂起运算而不会阻塞线程。阻塞线程通常代价昂贵,特别是在高负载的情况下,因为在真实的情境下,仅仅会很少的几个线程会在运行,所以阻塞其中一个线程会导致很多重要的工作会被延迟。

换句话说,Coroutine基本上是无代价的。不需要切换上下文,也不需要任何OS的参与。在这基础上,suspension可以在很大程度上被使用者library控制,作为一个library的开发者,我们可以决定suspension的时候做什么,比如优化/打印/拦截,完全可以根据我们自己的需要。

不同的地方是,Coroutines不能被随时挂起,仅仅是在调用函数的时候。

通过suspend定义一个挂起函数

suspend fun doSomething(foo: Foo): Bar {
    ...
}

在kotlinx.coroutines中有

fun <T> async(block: suspend () -> T)

所以我们可以如下调用

async {
    doSomething(foo)
    ...
}

但是我们需要注意,不能在一个普通的函数如main()中调用挂起函数

fun main(args: Array<String>) {
    doSomething() // ERROR: Suspending function called from a non-coroutine context 
}

而且挂起函数可以是虚函数,可以被子类覆写

interface Base {
    suspend fun foo()
}

class Derived: Base {
    override suspend fun foo() { ... }
}

使用@RestrictsSuspension注释可以使用者定义新的挂起方式,最经典的例子是

@RestrictsSuspension
@SinceKotlin("1.1")
public abstract class SequenceBuilder<in T> internal constructor() {
    /**
     * Yields a value to the [Iterator] being built.
     *
     * @sample samples.collections.Sequences.Building.buildSequenceYieldAll
     * @sample samples.collections.Sequences.Building.buildFibonacciSequence
     */
    public abstract suspend fun yield(value: T)

    /**
     * Yields all values from the `iterator` to the [Iterator] being built.
     *
     * The sequence of values returned by the given iterator can be potentially infinite.
     *
     * @sample samples.collections.Sequences.Building.buildSequenceYieldAll
     */
    public abstract suspend fun yieldAll(iterator: Iterator<T>)

    /**
     * Yields a collections of values to the [Iterator] being built.
     *
     * @sample samples.collections.Sequences.Building.buildSequenceYieldAll
     */
    public suspend fun yieldAll(elements: Iterable<T>) {
        if (elements is Collection && elements.isEmpty()) return
        return yieldAll(elements.iterator())
    }

因为SequenceBuilder类的成员函数都是挂起函数,设计者不希望使用者添加新的挂起函数,所以可以使用@RestrictsSuspension做了限制。

Coroutines的内部工作机制,这个比较复杂,使用是的编译器技巧,有兴趣可以看英文资料Coroutines

需要注意是的,Coroutines目前还处于实验阶段。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
根据提供的引用内容,出现了两个关于Kotlin的错误信息。第一个引用中显示了一个无法解析依赖的错误,指出无法下载kotlin-reflect.jar文件。第二个引用中显示了一个关于kotlin-gradle-1.8.10.jar (org.jetbrains.kotlin:kotlin-reflect)",这个错误通常是由于Gradle无法找到所需的kotlin-reflect库而引起的。解决这个问题的方法是确保你的项目的Gradle配置正确,并且指定了正确的Kotlin版本。 你可以尝试以下几个步骤来解决这个问题: 1. 确保你的项目的build.gradle文件中包含了正确的Kotlin版本和kotlin-gradle-plugin版本。你可以在build.gradle文件中找到类似于以下代码的部分: ```groovy ext { kotlin_version = '1.8.10' } dependencies { // ... implementation "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version" // ... } buildscript { // ... dependencies { // ... classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" // ... } } ``` 请确保kotlin_version变量的值与你想要使用的Kotlin版本一致,并且在dependencies和buildscript的classpath中正确引用了kotlin-gradle-plugin。 2. 如果你已经确认了build.gradle文件中的配置正确无误,那么可能是因为Gradle无法从远程仓库下载kotlin-reflect.jar文件。你可以尝试清除Gradle的缓存并重新构建项目。在命令行中执行以下命令: ```shell ./gradlew clean ``` 然后重新构建项目: ```shell ./gradlew build ``` 这将清除Gradle的缓存并重新下载所需的依赖。 3. 如果上述步骤***切换到其他网络环境来解决这个问题。 希望以上步骤能够帮助你解决问题。如果你还有其他问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值