为什么要使用 .asStateFlow() 而不是直接赋值?

记录开发过程中疑问,如果说的不对,欢迎提出修改

这里我使用了两个MutableStateFlow和两个StateFlow做说明,其中strList是由_strList.asStateFlow()赋值,而str2List_strList直接赋值。这两种在代码运行都能正常运行,这也是为什么我对此有疑问,从而做下笔记梳理

示例代码及运行结果

kotlin 复制代码
private val _strList = MutableStateFlow<List<String>>(listOf("A", "B", "C"))
val strList: StateFlow<List<String>> = _strList.asStateFlow()

private val _str2List = MutableStateFlow<List<String>>(listOf("X", "Y", "Z"))
val str2List: StateFlow<List<String>> = _str2List

@Test
fun testAsStateFlow() = runTest {
    launch {
        strList.collect { println("strList 收到新值: $it") }
    }
    launch {
        delay(500) 
        (strList as? MutableStateFlow)?.value = listOf("A", "B", "C", "D")
    }
    launch {
        str2List.collect { println("str2List 收到新值: $it") }
    }
    launch {
        delay(500)
        (str2List as? MutableStateFlow)?.value = listOf("A", "X", "Y", "Z")
    }
}

下面是运行结果,可以看出通过.asStateFlow() 赋值的strList 无法直接通过asStateFlow转换从而修改参数,而str2List却可以

strList 收到新值: [A, B, C]

str2List 收到新值: [X, Y, Z]

str2List 收到新值: [A, X, Y, Z]

1. 什么是 MutableStateFlowStateFlow

  • MutableStateFlow 是一种可变的状态流,它允许我们通过 value 属性修改流中的状态。
  • StateFlow 是一种不可变的状态流,它只暴露给外部读取状态的能力,而不能修改状态。

通常,在封装状态时,我们将 MutableStateFlow 定义为私有属性(例如 _strList),而将其暴露为不可变的 StateFlow(例如 strList),这样外部代码只能读取状态,而无法修改它。

2. 为什么不建议直接赋值?

在上述代码中,如果直接将 _strList_str2List 作为 StateFlow 来暴露,会导致外部代码可以通过强制转换或直接访问 _strList_str2List 来修改其值。这样可能会破坏状态的不可变性,导致不可预测的副作用或状态变更。

例如,在以下代码中,尝试直接修改 strList 的值:

ini 复制代码
(strList as? MutableStateFlow)?.value = listOf("A", "B", "C", "D")

虽然 strListStateFlow 类型,它不能直接修改,但通过强制转换为 MutableStateFlow 类型,代码可以绕过不可变性并直接修改 value 属性。这违背了状态不可变性的设计原则,容易导致不可控的状态更改。

3. asStateFlow 的作用

使用 asStateFlow() 方法可以将 MutableStateFlow 转换为不可变的 StateFlow。这样,外部代码只能读取状态,而无法修改它。通过这种方式,我们可以确保状态的不可变性并防止外部代码直接更改状态。

swift 复制代码
val strList: StateFlow<List<String>> = _strList.asStateFlow()

这样,外部代码只能使用 strList.collect { ... } 来收集状态的变化,而不能修改 strList 的值。这使得状态管理更加安全和可控。

4. 总结

  • MutableStateFlow 提供了修改状态的能力,而 StateFlow 只允许读取状态。
  • 使用 asStateFlow() 可以将 MutableStateFlow 转换为不可变的 StateFlow,从而保护状态不被外部代码直接修改。
  • 通过这种方式,我们可以确保状态的不可变性,减少副作用,使代码更加健壮和可维护。

5.延伸问题:为什么要使用 asSharedFlow

类似于 StateFlowSharedFlow 也有可变和不可变的版本。MutableSharedFlow 允许外部代码通过 emit() 向流中发送事件,而 SharedFlow 则是一个只读流,只能被外部代码订阅和收集,不能直接修改流中的内容。

为了确保流的不可变性,应该使用 asSharedFlow() 方法将 MutableSharedFlow 转换为不可变的 SharedFlow。这样,外部代码只能订阅事件流,而不能通过 emit() 向流中发送事件,避免了意外修改流内容的副作用。

相关推荐
yeziyfx2 小时前
kotlin中 ?:的用法
android·开发语言·kotlin
Kapaseker14 小时前
千锤百炼写View 摸爬滚打名声就
android·kotlin
zFox1 天前
四、ViewModel + StateFlow + 状态持久化
kotlin·stateflow·viewmodel
冬奇Lab1 天前
【Kotlin系列03】控制流与函数:从if表达式到Lambda的进化之路
android·kotlin·编程语言
zh_xuan1 天前
kotlin 类继承的语法
开发语言·kotlin
冬奇Lab2 天前
【Kotlin系列02】变量与数据类型:从val/var到空安全的第一课
android·kotlin·编程语言
城堡修炼者2 天前
lazycolumn无动态刷新时【自看】
android·kotlin
资生算法程序员_畅想家_剑魔2 天前
Kotlin常见技术分享-02-相对于Java 的核心优势-协程
java·开发语言·kotlin