kotlin协程

kotlin协程的启动模式 (枚举类CoroutineStart)

CoroutineStart.DEFAULT

*Default——根据协程的上下文立即调度协程执行。* *如果coroutine上下文的[CoroutineDispatcher]从[CoroutineDispatcher]返回' true '。和大多数调度程序一样,
isdispatchrequired] 
*函数的作用是:然后将协调程序代码分派给以后执行,而调用协调程序构建器的代码将继续执行。
* *注意[调度员。unrestricted总是从它的[CoroutineDispatcher]返回' false '。函数,因此使用[Dispatchers启动一个协程。默认情况下unrestricted和使用un分派是一样的。
* *如果coroutine [Job]在有机会开始执行之前就被取消了,那么它根本不会开始它的
*执行,而是会在异常情况下完成。协调程序在挂起点的可取消性取决于
*挂起函数的特定实现细节。使用
[suspend endcancellablecoroutine]实现可取消的挂起函数。

CoroutineStart.LAZY

只有在需要时才懒洋洋地启动协同程序。
*
*有关详细信息,请参阅相应的协程构建器的文档
*(如[发射][CoroutineScope。发射]和[异步][CoroutineScope.async])。
*
*如果coroutine [Job]在有机会开始执行之前就被取消了,那么它将不会开始执行
*完全执行,但会在例外情况下完成。

CoroutineStart.ATOMIC 1.3.31以前试验阶段

自动(即。,以不可取消的方式)根据协程的上下文安排协程的执行。
*这类似于[默认值],但是协程在开始执行之前不能被取消。
*
*协调程序在暂停点的可取消性取决于特定的实现细节
*暂停函数,如[默认]。

CoroutineStart.UNDISPATCHED 1.3.31以前试验阶段

立即执行协程,直到它的第一个挂起点_in current thread_,就像
*协同程序使用[dispatcher . unrestricted]启动。但是,当协程从暂停状态恢复时
*根据上下文中的[CoroutineDispatcher]进行调度。
*
*这与[ATOMIC]类似,在某种意义上,协程开始执行,即使它已经被取消,
但是不同的是,它开始在同一个线程中执行。
*
*协调程序在暂停点的可取消性取决于特定的实现细节
*暂停函数,如[默认]。
*
** *注意:这是一个实验性的api。在未来使用这种模式时,协程的执行语义可能会发生变化。

job.join() 等待job执行完, 再执行后面的代码
CoroutineStar.DEFAULT模式(如果启动之前cancel, 协程里面的代码执行不到)

fun main() = runBlocking {

    val job: Job = GlobalScope.launch(start = CoroutineStart.DEFAULT) {
        println("1")
        delay(1000L)
        println("2")
    }
//    job.start()
//    job.join()
    println("3")
    delay(2000L)//为了保证结果都能打印, 因为外面的协程1, 不会等待里面的协程2执行完.
}
打印结果, 不加delay(2000L)时:
3
1
打印结果, 加上delay(2000L)时:
3
1
2

CoroutineStart.LAZY模式

fun main() = runBlocking {//协程1

    val job: Job = GlobalScope.launch(start = CoroutineStart.LAZY) {//协程2
        println("1")
        delay(1000L)
        println("2")
    }
    job.start()
    job.join() //等待job执行完, 再执行后面的代码
    println("3")
}
打印结果如下:
1
2
3

Deferred

private suspend fun test2() {
    val deferred1 = GlobalScope.async<Int>(start = CoroutineStart.LAZY) {
        delay(1000L)
        println("计算")
        2
    }
    val deferred2 = GlobalScope.async<Int>(start = CoroutineStart.LAZY) {
        delay(1000L)
        println("计算2")
        3
    }

    val num = deferred1.await() + deferred2.await()//等待deferred1和deferred2两个协程执行完

    println(num)
}


CoroutineStart.DEFAULT和CoroutineStart.ATOMIC区别

suspend fun test3() {
    val job = GlobalScope.launch (start = CoroutineStart.DEFAULT){
        println("1")
        delay(1000L)
        println("2")
    }
    //默认模式 如果执行之前被取消, 里面的内容可能执行不到
    job.cancel()
    println("3")

    val job2 = GlobalScope.launch (start = CoroutineStart.ATOMIC){
        println("4")
        //第一个挂起点
        delay(1000L)
        println("5")
    }
    //自动模式 如果只要执行了, 在第一个挂起点取消才会生效, 也就是4一定会被打印
    job2.cancel()
    println("6")
}

CoroutineStart.UNDISPATCHERED模式

fun main() = runBlocking<Unit> {
    test4()
}

suspend fun test4() {
    //默认和父协程在同一个线程
    val job = GlobalScope.launch(start = CoroutineStart.UNDISPATCHED){
        println("1")
        delay(1000)
        println("2")
    }
    println("5")
    //等待job执行完成, 再执行后面的逻辑
    job.join()
}

协程的调度

协程调度器类型

2019-06-02 at 1.58 PM.png

协程调度器实际上是一个协程拦截器

2019-06-02 at 1.51 PM.png

自定义ContinuationInterceptor和Continuation

fun log(str:String){
    println("${Thread.currentThread().name}, $str")
}
suspend fun main()  {
    val job = GlobalScope.launch(MyContinuationInterceptor() + CoroutineName("HelloWorld")) {
        log("1")
        delay(1000L)
        log("2")
    }

    job.join()
}

class MyContinuationInterceptor : ContinuationInterceptor {
    //ContinuationInterceptor 是ContinuationInterceptor类中的伴生对象
    override val key: CoroutineContext.Key<*> = ContinuationInterceptor
    override fun <T> interceptContinuation(continuation: Continuation<T>): Continuation<T> {
        return MyContinuation(continuation)
    }

}

class MyContinuation<T>(private val continuation: Continuation<T>) : Continuation<T> {

    override val context: CoroutineContext = continuation.context

    private val executor = Executors.newSingleThreadExecutor{ it ->
        Thread(it, "MyThreadExecutor").also { it.isDaemon = true }
    }

    override fun resumeWith(result: Result<T>) {

        executor.submit {
            log("Before...")
            continuation.resumeWith(result)
            log("After...")
        }
    }

}
打印结果:
MyThreadExecutor, Before...
MyThreadExecutor, 1
MyThreadExecutor, After...
MyThreadExecutor, Before...
MyThreadExecutor, 2
MyThreadExecutor, After...

协程的异常处理

协程的异常处理


fun main() = runBlocking<Unit> {
    test11()
    test22()
    test33()
}

private suspend fun test33() {
    GlobalScope.launch {
        val job = launch {
            val num = 3 / 0 //此处发生了异常, 导致父协程被取消, 从而引起job.join()触发已经被取消异常
            delay(1000L)
        }

        try {
            job.join()
        } catch (e: Exception) {//已经被取消异常
            println("test33: $e.localizedMessage")
        }
    }.join()
}

private suspend fun test22() {
    val deferred = GlobalScope.async {//async自己会捕获内部的异常
        val num = 2 / 0
        delay(1000L)
        3
    }

    try {
        val i = deferred.await()
    } catch (e: Exception) {
        println("test22: $e.localizedMessage")
    }
}

private suspend fun test11() {
    val job = GlobalScope.launch(CoroutineExceptionHandler { _, throwable ->
        println("test11: $throwable.localizedMessage")
    }) {//launch 自己不会捕获内部异常
        val num = 3 / 0
        delay(1000L)
    }
    job.join()
}

协程的异常处理2

val demo = Demo1()

fun main() = runBlocking<Unit> {
    demo.test()
}

class Demo1 {
    fun test() = runBlocking {
        GlobalScope.launch(CoroutineExceptionHandler { _, throwable ->
            println("#1 ${throwable.localizedMessage}")
        }) {
            println("1")
            val job = launch(CoroutineExceptionHandler { _, throwable ->
                println("#2 ${throwable.localizedMessage}")
            }) {
                println("2")
                val num = 2 / 0
                println("2 after...")
            }
            println("3")
            job.join()
            println("4")
        }.join()
    }
}

打印结果:
1
3
2
#1 / by zero
异常被父协程捕获, 异常层层向上传递

使用suspervisorScope{}包裹子协程, 子协程自己捕获异常


val demo = Demo1()

fun main() = runBlocking<Unit> {
    demo.test2()
}

class Demo1 {
    fun test2() = runBlocking {
        GlobalScope.launch(CoroutineExceptionHandler { _, throwable ->
            println("#1 ${throwable.localizedMessage}")
        }) {
            println("1")
            supervisorScope {//不想外部传递异常, 但是外部父协程可以取消此\job
                val job = launch(CoroutineExceptionHandler { _, throwable ->
                    println("#2 ${throwable.localizedMessage}")
                }) {
                    println("2")
                    val num = 2 / 0
                    println("2 after...")
                }
            }
            println("3")
        }.join()
    }
}

打印结果:
1
2
#2 / by zero
3

coroutineScope{}继承外部协程上下文环境

val demo = Demo1()

fun main() = runBlocking<Unit> {
    demo.test3()
}
 fun test3() = runBlocking {
        GlobalScope.launch(CoroutineExceptionHandler { _, throwable ->
            println("#1 ${throwable.localizedMessage}")
        }) {
            println("1")
            test4()
            println("3")
        }.join()
    }

    private suspend fun test4(){//继承外部协程环境
        coroutineScope {
            val job = launch(CoroutineExceptionHandler { _, throwable ->
                println("#2 ${throwable.localizedMessage}")
            }) {
                println("2")
                val num = 2 / 0
                println("2 after...")
            }
        }
    }
打印结果:
1
2
#1 / by zero

协程作用域与异常传播

2019-06-02 at 8.24 PM.png

异常捕获处理机制

2019-06-02 at 8.30 PM.png

协程的取消

使用协程封装Retrofit请求数据

    private fun test3() {
        GlobalScope.launch {
            val json = getResult("/adult/user/logon", mutableMapOf("mobile" to "18795455272", "valid_code" to "8158"))
            Log.e("test:: ", json)//打印结果
        }
    }

    private suspend fun getResult(url: String, params: MutableMap<String, String>): String =
            suspendCancellableCoroutine<String> { continuation ->
                val call = RetrofitUtils.getApiService().post(url, params)
                //失败取消请求
                continuation.invokeOnCancellation { call.cancel() }
                call.enqueue(object : retrofit2.Callback<JsonObject> {
                    override fun onFailure(call: Call<JsonObject>, t: Throwable) {
                        continuation.resumeWithException(t)
                    }

                    override fun onResponse(call: Call<JsonObject>, response: Response<JsonObject>) {

                        if (response.isSuccessful) {
                            response.body()?.toString()?.let { continuation.resume(it) }
                                    ?: continuation.resumeWithException(NullPointerException("json is null"))

                        } else {
                            continuation.resumeWith(Result.failure(HttpException(response)))
                        }
                    }

                })
            }

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

推荐阅读更多精彩内容

  • 前言  今年的Google开发者大会已表明将Kotlin作为其正式的语言,现Google大力主推Kotlin, 在...
    Vgecanshang阅读 3,445评论 0 15
  • 轻量级线程:协程 在常用的并发模型中,多进程、多线程、分布式是最普遍的,不过近些年来逐渐有一些语言以first-c...
    Tenderness4阅读 6,351评论 2 10
  • 关键词:Kotlin 协程 启动模式 现在你已经知道协程大概是怎么回事了,也应该想要自己尝试一把了吧。本文将为大家...
    Kotlin中文社区阅读 3,524评论 5 13
  • 关键词:Kotlin 异步编程 协程 上一篇我们知道了协程启动的几种模式,也通过示例认识了 launch 启动协程...
    Kotlin中文社区阅读 3,769评论 5 11
  • 这几年心里总有一团乌云压在那里,见不阳光,郁郁寡欢。都快忘了什么时候真正的开怀笑过。 那时候好像还是大学,如果重新...
    梓岭金草阅读 352评论 0 2