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;要同步态 → 读变量或快照,别只盯流。

相关推荐
Tian_Hang14 小时前
C++原型模式(Protype)
开发语言·c++·算法
天天讯通14 小时前
OKCC 呼叫中心安全性能全解析:技术防护与管理措施指南
大数据·开发语言·网络·人工智能·安全·语音识别
xufengzhu14 小时前
第三方 Python 库 redis-py + hiredis 的使用
开发语言·redis·python
jingling55514 小时前
go | 环境安装和快速入门
开发语言·后端·golang
yuan1999715 小时前
欧拉梁静力与屈曲计算的 MATLAB 实现(有限差分法 + 解析解)
开发语言·算法·matlab
llxxyy卢15 小时前
polar夏季赛部分题目
开发语言·python
AI玫瑰助手15 小时前
Python模块:from...import...导入指定内容
开发语言·python·信息可视化
石山代码15 小时前
JavaScript 进阶核心知识点
开发语言·javascript·ecmascript
FL162386312915 小时前
[cmake]基于C++使用纯opencv部署ppocrv5v6的onnx模型
开发语言·c++·opencv