Kotlin創建一個好用的協程作用域

前言

kotlin中使用協程,是一定要跟協程作用域一起配合使用的,否則可能協程的生命周期無法被準確控制,造成內存泄漏或其他問題.

我們一般在安卓項目中使用協程作用域,可能會在BaseActtivity中new 一個MainScope(),並在onDestory時cancel掉,或者隻在ViewModel中使用viewModelScope,然後會在ViewModel的onClose中自動cancel掉.

但我們可能不隻需要這些效果,比如在協程作用域中拿到Context或Activity,或者需要統一的捕獲異常,下面我們就來探討一下如何創建一個好用的作用域.

正文

首先我們自己創建協程作用域需要調用CoroutineScope()方法,然後通過加號+拼接CoroutineContext

首先我們需要確認我們要什麼Job,如果需要連帶責任的(子協程取消,父協程也會取消),就使用Job().如果隻能由父控制子的取消,就使用SupervisorJob()

然後確定我們的協程作用域是要默認運行在哪個線程中,一般安卓開發都是默認主線程,這裡我們使用Dispatchers.Main.immediate,Dispatchers.Main.immediate和Dispatchers.Main的區別就是,Dispatchers.Main會直接post到主線程,而Dispatchers.Main.immediate如果發現自身是在主線程,就可能不再post一次,而是直接調度

然後我們可能需要對協程作用域中的異常進行統一處理,所以我們附加上一個異常處理器協程上下文元素:

/**
 * 設置協程異常策略的上下文元素
 */
object CoroutineExceptionHandlerWithReleaseUploadAndDebugThrow
    : AbstractCoroutineContextElement(CoroutineExceptionHandler), CoroutineExceptionHandler {
    override fun handleException(context: CoroutineContext, exception: Throwable) {
        if (exception !is CancellationException) {//如果是SupervisorJob就不會傳播取消異常,而Job會傳播
            //todo 處理異常
        }
    }
}

我們為瞭方便調試,需要給協程作用域附加上名稱,這裡我們使用自帶的協程上下文元素CoroutineName()

而我們可能需要通過協程作用域取到安卓的Context或者Activity等,這裡我們也可以自定義一些協程上下文元素,如:

/**
 * 存儲BaseActivity的協程屬性
 */
class CoroutineElementWithBaseActivity(val baseActivity: BaseActivity) :
    AbstractCoroutineContextElement(CoroutineElementWithBaseActivity) {
    companion object Key : CoroutineContext.Key<CoroutineElementWithBaseActivity>
}

這樣我們可以在有協程作用域的地方取到activity並作出彈加載窗等操作:

完整代碼如下:

到此這篇關於Kotlin創建一個好用的協程作用域的文章就介紹到這瞭,更多相關Kotlin創建協程作用域內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: