Android:坑-协程-订阅时序导致收不到流

坑-协程-订阅时序导致收不到流

封装了一个MVI框架,极低概率出现这样一种问题:

页面A、页面B,A跳转B,B再返回A,然后A立刻再跳入B,偶现ViewModelScope.launch{} 域中的ShareFlow没有触发。

代码:

kt 复制代码
abstract class BaseFlowViewModel<VS : ViewState, I : UserIntent,Changes: ReduceViewStateChange<VS>>(
    application: Application,
    firstViewState: VS
) : BaseViewModel(application) {

    private val _viewState: MutableStateFlow<VS> = MutableStateFlow(firstViewState)
    val viewState : MutableStateFlow<VS> = _viewState

    private val _intent:MutableSharedFlow<I> = MutableSharedFlow(extraBufferCapacity = 5, onBufferOverflow=BufferOverflow.DROP_OLDEST)

    protected abstract suspend fun realSendIntent(intent: I)

    protected abstract suspend fun catchError(throwable: Throwable)

    fun sendIntent(intent: I){
        Log.d("wuxu","sendIntent!!!")
        viewModelScope.launch {
            _intent.emit(intent)
        }
    }

    init {
        viewModelScope.launch(Dispatchers.IO+SupervisorJob()) {
            Log.d("wuxu","init!!!")
            _intent
                .onEach {
                    Log.d("wuxu","viewModelScope onEach $it")
                    realSendIntent(it)
                }
                .catch {
                    catchError(it)
                }
                .collect()
        }
    }
}

猜想原因只能有三个:

  1. scope被cancel;- 通过源码及日志输出判断,每次页面进入都会创建新的ViewModelScope,不存在cancel的情况;
  2. 协程域异常;- 这个我使用的是SupervisorJob(),所以排除
  3. collect()订阅,比emit晚。 - 最终确定是这个

最终日志输出:

正常:init->sendIntent 出问题时:sendIntent->init ,此时没有collect,导致无法触发流。

相关推荐
野生的码农5 小时前
码农的妇产科实习记录
android·java·人工智能
王正南5 小时前
kali-linux 虚拟机连接安卓模拟器
android·linux·运维·虚拟机连接模拟器·安卓模拟器,linux虚拟机
撩得Android一次心动5 小时前
Android Jetpack 概述
android·android jetpack
JinBeen5 小时前
sourcetree下码云gitee的ssh经常失效问题
android·gitee·ssh
帅得不敢出门6 小时前
Android各芯片平台日志打开及获取
android
极客小云6 小时前
【Android Gradle 构建常见报错及解决方法大全】
android·运维开发
Just_Paranoid7 小时前
【TaskbarDelegate】屏蔽上滑返回桌面手势功能
android·systemui·navigation·launcher·gesture
赛恩斯7 小时前
asfp 如何导入并使用aosp13
android
诸神黄昏EX7 小时前
Android Safety 系列专题【篇三:Secure Boot机制】
android
李坤林8 小时前
Android Binder 详解(4) Binder 线程池
android·java·binder