Skip to content

Kotlin协程入门

更新日期 2021-9-22
  • 2021-9-22 创建文档
开发环境
  • IntelliJ IDEA 2021.2.2 (Community Edition)
  • Kotlin: 212-1.5.10-release-IJ5284.40

Kotlin协程系列目录

  1. Kotlin协程入门 —— 第一个例子,启动协程
  2. Kotlin协程基础 —— 阻塞与非阻塞,等待,作用域构建器
  3. Kotlin协程取消与超时 —— 取消的条件,超时设定


介绍Kotlin中的协程。用一个例子来展示协程的基本用法。

第一个例子

新建工程

我们使用的是社区版IntelliJ IDEA 2021.2.2。新建一个Kotlin工程用来测试。

新建工程

新建工程

引入协程

项目用gradle进行管理,我们在Github上找到协程的依赖。

dependencies {
    implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.5.2")
}

修改项目的gradle配置。把依赖添加进去。

gradle

gradle引入协程

代码示例

创建一个类然后在main方法中写协程的相关代码。

import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch

fun main() {
    GlobalScope.launch { // 在后台启动一个新的协程并继续
        delay(300)       // 等待300毫秒
        "rustfisher.com".forEach {
            print(it)
            delay(200)   // 每次打印都等待一下
        }
    }
    println("RustFisher")
    Thread.sleep(3000) // 阻塞主线程防止过快退出
}
代码运行结果

运行效果

本质上,协程是轻量级的线程。

我们用GlobalScope启动了一个新的协程,这意味着新协程的生命周期只受整个应用程序的生命周期限制。

可以将 GlobalScope.launch { …… } 替换为 thread { …… },并将 delay(……) 替换为 Thread.sleep(……) 达到同样目的。注意导入包kotlin.concurrent.thread

协程换成线程
import java.lang.Thread.sleep
import kotlin.concurrent.thread

fun main() {
    thread {
        sleep(300)
        "rustfisher.com".forEach {
            print(it)
            sleep(200)
        }
    }
    println("RustFisher")
    sleep(3000) // 阻塞主线程防止过快退出
}

如果thread{}中含有delay,编译器会报错

delay报错

Suspend function 'delay' should be called only from a coroutine or another suspend function

因为delay是一个特殊的挂起函数,它不会造成线程阻塞,但是会挂起协程,并且只能在协程中使用。

至此,小结一下第一个协程示例

  • gradle引入协程kotlinx-coroutines-core
  • GlobalScope.launch启动协程
  • 协程中的挂起函数delay(long)可以达到延时的效果,并且它只能在协程中使用

协程所在线程

本质上协程是轻量级的线程。我们观察一下协程所处的线程信息。

观察线程信息
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.launch
import java.lang.Thread.sleep

fun main() {
    println("main线程信息 ${Thread.currentThread().id}")
    for (i in 1..3) { // 多启动几次协程
        GlobalScope.launch {
            println("协程启动#$i 所在线程id: ${Thread.currentThread().id}")
        }
    }
    sleep(2000) // 阻塞主线程防止过快退出
    println("RustFisher 示例结束")
}
运行log如下
main线程信息 1
协程启动#1 所在线程id: 11
协程启动#3 所在线程id: 14
协程启动#2 所在线程id: 13
RustFisher 示例结束

可以看到多次启动协程,这些协程它们不一定在同一个线程中。也就是说有在同一个线程的可能性。

于是试试疯狂地启动协程,把循环次数加大for (i in 1..3000)。观察log可以看出,有重复的线程id。

入门思维导图

作者: rustfisher.com | rf.cs@foxmail.com
示例: AndroidTutorial Gitee, Tutorial Github
本文链接: https://www.an.rustfisher.com/kotlin/coroutines/basic/first/
一家之言,仅当抛砖引玉。如有错漏,还请指出。如果喜欢本站的内容,还请支持作者