Kotlin 协程面试深度解析:coroutineScope 与 supervisorScope

前言:

在软件开发的面试中,经常会遇到涉及并发编程和异常处理的问题。面试官可能会问:“如果你需要同时执行多个任务,但只要其中一个任务失败,就希望立即中止其他任务,你会如何处理?”这个问题旨在考察面试者对于并发编程和异常处理的理解。 在 Java 中,JUC是处理并发编程的主要工具之一。而在 Kotlin 中,协程是一种强大的并发编程工具,提供了更加简洁和灵活的方式来处理并发任务。

本文将首先使用 Java JUC 来实现解决这个问题的方案,然后再深入探讨 Kotlin 中的 coroutineScopesupervisorScope 的区别。通过对比两种不同的实现方式,读者可以更清晰地理解 Kotlin 协程的优势所在。

让我们从面试题的情景开始,先看看如何使用 Java JUC 来解决这个问题。

Java JUC 与 Kotlin 协程

首先,我们将介绍 Java JUC和 Kotlin 协程。JUC 在 Java 中是处理并发编程任务的主要工具之一,而 Kotlin 协程则提供了一种更简洁、灵活的方式来处理并发任务。

使用 Java JUC 实现并发任务

在 Java 中,可以使用 Java Util Concurrency(JUC)来处理并发编程任务。下面是一个使用 Java JUC 实现并发任务的示例代码:

import kotlinx.coroutines.*
import java.util.concurrent.CountDownLatch
import java.util.concurrent.atomic.AtomicBoolean

 fun multipleTaskUseJuc() {
    val latch = CountDownLatch(5) // 初始化计数器为 5
    val hasFailed = AtomicBoolean(false) // 标志是否有任务失败,默认为 false

    val taskStart = System.currentTimeMillis()
    for (i in 1..5) {
        launch {
            doSomethingAsync(i, object : AsyncCallback {
                override fun onSuccess(result: String) {
                    println(result)
                    latch.countDown() // 减少计数
                    checkForFailure()
                }

                override fun onFailure(error: Throwable) {
                    println("Task failed: ${error.message}")
                    latch.countDown() // 减少计数
                    hasFailed.set(true) // 标记任务失败
                    checkForFailure()
                }

                fun checkForFailure() {
                    if (hasFailed.get()) {
                        println("Task failed: One or more tasks failed")
                        // 取消其他任务
                        repeat(latch.count.toInt()) {
                            latch.countDown()
                        }
                    } else if (latch.count == 0L) {
                        println("All tasks completed successfully")
                    }
                }
            })
        }
    }
    latch.await() // 等待所有任务完成或有任务失败
    println("end--> cost:::${System.currentTimeMillis() - taskStart}")
}

 fun doSomethingAsync(taskId: Int, callback: AsyncCallback) {
    // 模拟异步操作
    Thread {
        try {
            Thread.sleep((2000 * taskId).toLong()) // 模拟耗时操作
            if (taskId == 3) {
                throw Exception("Task $taskId failed")
            }
            callback.onSuccess("Task $taskId -->completed successfully")
        } catch (e: InterruptedException) {
            callback.onFailure(e)
        } catch (e: Exception) {
            callback.onFailure(e)
        }
    }.start()
}

interface AsyncCallback {
    fun onSuccess(result: String)
    fun onFailure(error: Throwable)
}

在这段代码中,使用了 CountDownLatchAtomicBoolean 来处理任务完成和失败的情况。doSomethingAsync 函数模拟了异步任务的执行,如果任务 ID 为 3,则抛出异常。

接下来,我们将探讨如何使用 Kotlin 协程来实现同样的功能,并比较两种实现方式的优劣。

Kotlin 协程的异常处理与并发管理

在 Kotlin 中,协程是一种强大的并发编程工具,它可以极大地简化异步编程的复杂性。在处理并发任务时,coroutineScopesupervisorScope 是两个重要的概念。接下来,我们将深入探讨这两种作用域的使用方法以及它们之间的区别。

代码示例:使用 coroutineScope 实现并发任务

import kotlinx.coroutines.*

suspend fun multipleTaskUseCoroutineScope() {
    println("multipleTaskUseCoroutine start")
    val taskStart = System.currentTimeMillis()
    val taskId = mutableListOf(1, 2, 3, 4, 5)
    try {
        coroutineScope {
            taskId.map {
                launch {
                    println(doSomeAsync(it))
                }
            }
        }

    } catch (e: Exception) {
        e.printStackTrace()
        println("multipleTaskUseCoroutine failed:::${System.currentTimeMillis() - taskStart}")
    }
    println("multipleTaskUseCoroutine end--> cost:::${System.currentTimeMillis() - taskStart}")
}

suspend fun doSomeAsync(task: Int) = suspendCancellableCoroutine<String> { continuation ->
    doSomethingAsync(task, object : AsyncCallback {
        override fun onSuccess(result: String) {
            continuation.resume(result)
        }

        override fun onFailure(error: Throwable) {
            continuation.resumeWithException(error)
        }

    })
}


在这段代码中,我们使用了 coroutineScope 来创建一个协程作用域,并在其中启动了多个子协程来执行并发任务。doSomeAsync 函数模拟了异步任务的执行,如果任务 ID 为 3,则抛出异常。剩余的任务将会被取消。

接下来,我们将通过比较 coroutineScopesupervisorScope 的使用方法和特性来更深入地理解它们之间的区别。

共同点

首先,我们来看一下 coroutineScopesupervisorScope 的共同特性:

  • 作用域嵌套:两者均可以嵌套在其他作用域中。
  • 结构化并发:它们都强制执行结构化并发,即作用域会等待所有启动的子协程完成之后才会完成自身。
  • 取消传播:从父协程向子协程传播取消信号。如果父协程被取消,所有子协程也会被取消。

不同点

coroutineScopesupervisorScope 主要区别在于它们对异常处理和异常传播的方式:

  • 异常传播

    • coroutineScope:任何子协程抛出的异常都会立即取消整个作用域及其内的其他子协程。
    • supervisorScope:子协程抛出的异常不会导致整个作用域或其他子协程的立即取消。失败的协程将独立处理,允许其他协程继续执行。
  • 使用场景

    • coroutineScope 非常适合于一组相关任务,如果其中一个任务失败,其他任务也应该一同取消。
    • supervisorScope 在你希望独立处理子协程的失败并保持其他任务运行时非常有用,例如启动多个独立的操作。

代码示例:coroutineScope

以下是使用 coroutineScope 的代码示例:

import kotlinx.coroutines.*

fun main() = runBlocking {
    try {
        coroutineScope {
            launch {
                throw Exception("子协程失败")
            }
            launch {
                delay(100)
                println("因为其他子协程失败,这段代码不会被打印。")
            }
        }
    } catch (e: Exception) {
        println("捕获异常: ${e.message}")
    }
}

输出结果:

<TEXT>
捕获异常: 子协程失败

在上面的示例中,第一个子协程抛出了异常,这导致整个 coroutineScope 取消,包括第二个子协程,在它有机会打印消息之前就被取消了。

代码示例:supervisorScope

现在我们来看看 supervisorScope 的实际使用:

<KOTLIN>
import kotlinx.coroutines.*

fun main() = runBlocking {
    supervisorScope {
        launch {
            throw Exception("子协程失败")
        }
        launch {
            delay(100)
            println("尽管其他子协程失败了,这段代码仍然会被打印。")
        }
    }
}

输出结果:

<TEXT>
尽管其他子协程失败了,这段代码仍然会被打印。
主线程中的异常: java.lang.Exception: 子协程失败

在这个示例中,第一个子协程失败了,但第二个子协程继续运行并打印了消息。supervisorScope 允许独立处理失败协程的异常。

结论

理解 coroutineScopesupervisorScope 之间的区别对于编写高效的 Kotlin 协程代码至关重要。当你希望所有子协程因一个任务的失败而一同失败时,使用 coroutineScope;当你想独立处理子协程的失败并保持其他操作运行时,使用 supervisorScope。根据你的使用场景选择正确的作用域,你可以创建出更加健壮和响应迅速的应用程序。

最后

如果想要成为架构师或想突破20~30K薪资范畴,那就不要局限在编码,业务,要会选型、扩展,提升编程思维。此外,良好的职业规划也很重要,学习的习惯很重要,但是最重要的还是要能持之以恒,任何不能坚持落实的计划都是空谈。

如果你没有方向,这里给大家分享一套由阿里高级架构师编写的《Android八大模块进阶笔记》,帮大家将杂乱、零散、碎片化的知识进行体系化的整理,让大家系统而高效地掌握Android开发的各个知识点。
img
相对于我们平时看的碎片化内容,这份笔记的知识点更系统化,更容易理解和记忆,是严格按照知识体系编排的。

欢迎大家一键三连支持,若需要文中资料,直接扫描文末CSDN官方认证微信卡片免费领取↓↓↓(文末还有ChatGPT机器人小福利哦,大家千万不要错过)

PS:群里还设有ChatGPT机器人,可以解答大家在工作上或者是技术上的问题
图片

  • 12
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值