Kotlin中的协程 - Scope

前言

Kotlin是一种在Java虚拟机上运行的静态类型编程语言,被称之为Android世界的Swift,在GoogleI/O2017中,Google宣布Kotlin成为Android官方开发语言

CoroutineScope

当我们创建一个协程的时候,都会需要一个CoroutineScope,它是协程的作用域,我们一般使用它的launch函数以及async函数去进行协程的创建

public fun CoroutineScope.launch(
    context: CoroutineContext = EmptyCoroutineContext,
    start: CoroutineStart = CoroutineStart.DEFAULT,
    block: suspend CoroutineScope.() -> Unit
): Job {
    val newContext = newCoroutineContext(context)
    val coroutine = if (start.isLazy)
        LazyStandaloneCoroutine(newContext, block) else
        StandaloneCoroutine(newContext, active = true)
    coroutine.start(start, coroutine, block)
    return coroutine
}

Kotlin中的协程 - CoroutineContext 中我们了解到了,在launch函数中 具有传递上下文的功能,从而生成了Job链让协程之间结构化,并且我们的CoroutineContext也是定义在CoroutineScope当中的

public interface CoroutineScope {
    /**
     * The context of this scope.
     * Context is encapsulated by the scope and used for implementation of coroutine builders that are extensions on the scope.
     * Accessing this property in general code is not recommended for any purposes except accessing [Job] instance for advanced usages.
     *
     * By convention, should contain an instance of a [job][Job] to enforce structured concurrency.
     */
    public val coroutineContext: CoroutineContext
}

常用函数

launch 启动一个协程代码块
async 返回值Deferred,表示一个延期返回的结果
coroutineContext 获取当前的coroutineContext
cancel 取消掉当前Scope所对应的协程

launch函数
launch通常用来创建一个协程,它是CoroutineScope的函数

public fun CoroutineScope.launch(
    context: CoroutineContext = EmptyCoroutineContext,
    start: CoroutineStart = CoroutineStart.DEFAULT,
    block: suspend CoroutineScope.() -> Unit
): Job {
    val newContext = newCoroutineContext(context)
    val coroutine = if (start.isLazy)
        LazyStandaloneCoroutine(newContext, block) else
        StandaloneCoroutine(newContext, active = true)
    coroutine.start(start, coroutine, block)
    return coroutine
}

这个函数主要做了以下几件事情:

  1. 将传入的context与当前Scopecontext进行运算,生成一个运算之后的context,通过newCoroutineContext函数,这个Context并不是最终协程的Context,而是它的父Context

2.如果是非lazy模式下创建一个Job,它是StandaloneCoroutine的对象,并且在父类AbstractCoroutine中将创建的Job与上面生成的context在进行plus运算,最后才会生成此协程的Context对象

public final override val context: CoroutineContext = parentContext + this

3.然后在start中,将当前协程的Job对象指定为ParentJob,形成了父子关联,这个操作是在JobSupportinitParentJobInternal函数中实现

    internal fun initParentJobInternal(parent: Job?) {
        val handle = parent.attachChild(this)
        parentHandle = handle
    }

整个过程中 Scope的工作,就是将Context进行传递,使协程之间存在父子结构化,使取消事件和异常行为进行关联

取消协程
Scope中也有函数可以对协程进行取消

val scope = object : CoroutineScope {
    override val coroutineContext: CoroutineContext = Job()
}

//通过Scope创建一个协程
val job = scope.launch() {
    Log.e("Mike", "parent job start ${this.coroutineContext[Job]}")
  //通过同样的Scope创建一个协程
    val chindJob = scope.launch() {
        Log.e("Mike", "child job start ${this.coroutineContext[Job]}")
        delay(5000)
        Log.e("Mike", "child job end ${this.coroutineContext[Job]}")
    }
    delay(4000)
    Log.e("Mike", "parent job end ${this.coroutineContext[Job]}")
}
Thread.sleep(1000)
job.cancel()
打印结果
parent job start
child job start
五秒后
child job end

上面的代码中,虽然内部的协程时在外部协程中进行创建的,但是内部协程并无法通过Job去进行取消,原因就在于并没有使用到传递的CoroutineScope,使两个Job之间没有结构化关系,所以互不影响

//job.cancel()
scope.cancel()

换成scope中的cancel就可以正常的取消,因为两个协程的Job均是ScopeJobChild

val job = scope.launch() {
    val chindJob = launch() {
        delay(5000)
    }
    delay(4000)
}
Thread.sleep(1000)
job.cancel()

这时候Job之间使用CoroutineScope传递的Context,使Job之间有了关联,所以可以一起取消

MainScope与GlobalScope

GlobalScope中是一个EmptyCoroutineContext,其中并没有Job对象,所以也无法通过GlobalScope去取消关联的协程,所以它是进程级别的Scope

public object GlobalScope : CoroutineScope {
    /**
     * Returns [EmptyCoroutineContext].
     */
    override val coroutineContext: CoroutineContext
        get() = EmptyCoroutineContext
}

因为Scope也是通过Job去取消的

public fun CoroutineScope.cancel(cause: CancellationException? = null) {
    val job = coroutineContext[Job] ?: error("Scope cannot be cancelled because it does not have a job: $this")
    job.cancel(cause)
}

MianScope是由SupervisorJob和主线程调度组成的

public fun MainScope(): CoroutineScope = ContextScope(SupervisorJob() + Dispatchers.Main)

表明是无法被Cancel掉的

private class SupervisorJobImpl(parent: Job?) : JobImpl(parent) {
    override fun childCancelled(cause: Throwable): Boolean = false
}
val scope = object : CoroutineScope {
    override val coroutineContext: CoroutineContext = Job()
}
scope.launch {
    val response = async {
        delay(5000)
        "response data"
    }.await()
    Log.e("Mike", "async end")
    MainScope().launch {
        Log.e("Mike", "MainScope launch ")
    }
}
MainScope().cancel()
}
打印结果
5秒后
async end
MainScope launch

欢迎关注Mike的简书

Android 知识整理

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容