android kotlin Flow:distinctUntilChangedBy + stateIn 的坑

场景

kotlin 复制代码
class Link(val key: String) {
    enum class Phase { OFF, ON, RETRY }
    var phase: Phase = Phase.OFF
}

var active: Link? = null
val revision = MutableStateFlow(0L)  // 连接层 state 变了就 revision++

fun onPhaseChanged() {
    revision.value = System.currentTimeMillis()
}

fun onLinkLost() {
    active?.phase = Link.Phase.RETRY   // 不换 Link 实例,只改内部 phase
    onPhaseChanged()
}

onLinkLost() 只改 active!!.phase = RETRY不换新 Link 实例

问题写法:distinct 后面接 stateIn

kotlin 复制代码
val linkFlow: StateFlow<Link?> = revision
    .map { active }
    .distinctUntilChangedBy { "${it?.key}-${it?.phase}" }
    .stateIn(scope, SharingStarted.WhileSubscribed(), null)

phase 从 ON 变 RETRY 时:

步骤 发生什么
revision emit 触发整条链
distinctUntilChangedBy key 从 k-ON 变为 k-RETRY放行
stateIn.value 新值仍是同一个 active 引用 → old == new不通知 collect

distinct 按逻辑 key 去重;StateFlow引用 去重------多挡一层。

kotlin 复制代码
linkFlow.collect { link ->
    showOnline(link?.phase == Link.Phase.ON)
}

collect 不重跑时,showOnline 不会更新;link?.phase 虽是 getter,lambda 根本没执行。

.value 与缓存是同一规则:

kotlin 复制代码
val sameRef = linkFlow.value  // 引用在
// phase 已 RETRY,但 StateFlow 不会为此再推一次

替代:shareIn(replay = 1)

kotlin 复制代码
val linkFlow: SharedFlow<Link?> = revision
    .map { active }
    .distinctUntilChangedBy { "${it?.key}-${it?.phase}" }
    .shareIn(scope, SharingStarted.WhileSubscribed(), replay = 1)

distinct 放行后,每次SharedFlow emit 一次,不因引用相同而跳过 ;活跃 collect 会再跑。

stateIn shareIn(replay = 1)
同引用、仅 phase 常不通知 通知
晚订阅首包 .value replay 最后一次 emit
同步读 .value replayCache.lastOrNull() 或读 active?.phase

replay = 0 时晚订阅要等下一次 revision,进界面可能一直没有首包。

实践注意、边界与自检

  • 必须 StateFlow + .value 且实例可变:stateIn 前改成不可变快照(如 data class Snap(val key: String?, val phase: Phase)),不要直接缓存 Link?
  • 求简单、同引用 phase 也要推:distinctUntilChangedBy + shareIn(replay = 1)
  • 同步读当前 phase:直接读 active?.phase,别假设热流一定会再 emit。
  • 只改 active 却不 revision++:distinct 上游无 tick,换 shareIn 也没用。
  • 订阅方 if (last === link) return:自行挡刷新,与算子无关。

自检:distinct 日志/key 已变而 UI 不动 → 查 stateIn 引用判等;晚订阅无首包 → 查 replay;要同步态 → 读变量或快照,别只盯流。

相关推荐
Wang ruoxi13 小时前
Pygame 小游戏——打砖块
开发语言·python·pygame
AI科技星13 小时前
全域数学公理:32维超球体投影、微观曲率与霍奇猜想的几何化证明
c语言·开发语言·网络·量子计算·agi
幸运小圣14 小时前
前端三种输入数据来源生成 worksheet(工作表)新手适用详细篇【SheetJS】
开发语言·前端·javascript
美狐美颜sdk14 小时前
直播APP开发如何实现美颜功能?低成本美颜SDK方案推荐
android·人工智能·ios·第三方美颜sdk·视频美颜sdk
ch.ju14 小时前
Java Programming Chapter 4——Construction method
java·开发语言
烤代码的吐司君14 小时前
面向对象编程(OOP)在 Python 中的实现——类、继承与特殊方法
开发语言·python
AI行业学习14 小时前
CC-Switch Windows + macOS 下载安装配置全流程
java·开发语言·人工智能·python
Lumbrologist14 小时前
【C++】零基础入门 · 第 3 节:条件判断(if、switch)
开发语言·c++·算法
l1t14 小时前
DeepSeek总结的使用实体-组件-系统和基于存在性处理进行Python编程简介
开发语言·python