EventBus → SharedFlow

EventBus → SharedFlow

老写法(Java + EventBus)

java 复制代码
// 发送事件
EventBus.getDefault().post(new LoginEvent(userId));

// 接收事件
@Subscribe(threadMode = ThreadMode.MAIN)
public void onLoginEvent(LoginEvent event) {
    updateUI(event.getUserId());
}

// 注册/解注册
@Override
protected void onStart() {
    super.onStart();
    EventBus.getDefault().register(this);
}

@Override
protected void onStop() {
    super.onStop();
    EventBus.getDefault().unregister(this);
}

问题在哪里

EventBus 通过反射查找 @Subscribe 注解方法,性能损耗虽不大但存在。事件的发送方和接收方之间完全无约束,编译期不检查事件类型。忘调 register/unregister 是常见 bug 来源。全局事件满天飞,难以追踪事件的流向。

新写法(Kotlin + SharedFlow)

kotlin 复制代码
// 单例事件总线
object EventBus {
    private val _events = MutableSharedFlow<Any>(replay = 0,
            extraBufferCapacity = 64)
    val events: SharedFlow<Any> = _events

    suspend fun post(event: Any) { _events.emit(event) }
}

// 发送
viewModelScope.launch { EventBus.post(LoginEvent(userId)) }

// 接收
viewLifecycleOwner.lifecycleScope.launch {
    viewLifecycleOwner.repeatOnLifecycle(Lifecycle.State.STARTED) {
        EventBus.events.collect { event ->
            when (event) {
                is LoginEvent -> updateUI(event.userId)
            }
        }
    }
}

一句话注意

SharedFlow 替代 EventBus 的最大好处是类型安全------when (event) 配合 sealed class 定义事件,IDE 会提示所有可能的事件类型,不会遗漏。而且不需要 register/unregister,协程取消就自动停止接收。

extraBufferCapacity = 64 是为了没有消费者时也能缓存事件(类似 EventBus 的 sticky)。如果不需要 sticky,设 replay = 0, extraBufferCapacity = 0 并在 emit 时检查是否有订阅者。


Java Android 老项目迁移系列,持续更新中。

相关推荐
带刺的坐椅1 小时前
Spring Boot → Solon 注解迁移实战指南:一张对照表说清楚
java·springboot·web·solon
用户3721574261352 小时前
Java 将一个 PPT 文档拆分为多个文件
java
人活一口气17 小时前
Spring Boot与AIGC的完美结合:从零搭建智能内容生成平台
java·spring boot·aigc
像我这样帅的人丶你还19 小时前
Java 后端详解(三):全局异常处理与 JPA 数据库映射
java·后端
NE_STOP19 小时前
vibe Coding -- 小项目实战
java
未秃头的程序猿1 天前
Java 26正式发布!这3个新特性,让代码量直接减半
java·后端·面试
37手游移动客户端团队1 天前
招聘-高级安卓开发工程师
android·客户端
用户298698530141 天前
Word 文档文本查找与替换的 Java 实现方案
java·后端
阿哉1 天前
Nacos 服务发现源码:藏在背后的两套事件机制,90%的人只讲了一半
java