是否应该将Compose中的状态切分

如题,为什么会有这种疑问呢?

我们知道Compose是通过重组来刷新UI的,当一个State发生改变时,其对应的可重组函数的Lambda会重新执行,而我们通常在设计State时都是将当前界面相关属 combine成一个整体的State,所以我们在修改State时,其实就是让整个State都发生了变化,基本会让整个界面都发生重组不会跳过。下面通过一个简单的例子:

Kotlin 复制代码
// 一个整合的State
data class XModelState(
    val name: String,
    val age: Int,
    val intro: String
    ...
)
Kotlin 复制代码
Column(
    modifier = Modifier
    .fillMaxSize()
    .padding(16.dp)
) {
    
    println("Column")

    Wrapper {
        Text(text = "Name is ${modelState.name}").also {
            println("Name changed")
        }
    }
    Wrapper {
        Text(text = "Age is ${modelState.age}").also {
            println("Age changed")
        }
    }
    Wrapper {
        Text(text = "Intro is ${modelState.intro}").also {
            println("Intro changed")
        }
    }

    Button(
        modifier = Modifier.fillMaxWidth(),
        onClick = {
            modelState = modelState.copy(
                name = "Dougie ${Random.nextInt(0, 10)}",
                age = age + 1
            )
        }
    ) {
        Text(text = "Update Model")
    }
}

以上代码中,当点击Update Model按钮来修改modelState时,3个Wrapper全部都发生重组,假设这个State其实还可能更大,有20多个属性,而当我们只修改 一部分相关的,很多相关的Composable应该skip re-compose。

Kotlin 复制代码
*       ----------------     -----------------      ---------------------
*       | Click Button |  -> | State changed |  ->  | trigger recompose |
*       ----------------     -----------------      ---------------------

为了避免这个情况,目前Compose也没提供相关的api,好像也只能通过分割状态了,对应文章的标题。

1. ViewModel层保留多个公开的State

在ViewModel combine的时候就将State分割并提供多个State叫个UI层使用,而在操作的时候,我们也只会改变其对应的小State,其他State不会变化, 所以Compose上也只会重组其对应的可重组Lambda,而其他的则会跳过重组。

2. 使用 derivedStateOf

在View层,依然是拿到整个大的State,我们通过增加内部属性并使derivedStateOf这个方法将State分割。

Kotlin 复制代码
val age by remember {
    derivedStateOf {
        modelState.age
    }
}
val intro by remember {
    derivedStateOf {
        modelState.intro
    }
}

如上,将age和intro分割出来,在点击按钮是,age+1,而intro不变,通过日志,发现只有Age Changed,所以intro的被跳过重组了。

在日常开发中,需求feature会越来越多,State也会随之而增加,以上两种方式虽然一定程度上能减少重组,但是也很大程度上增加了维护工作 和带来新的问题。

总结

在Compose上,UI的更新必然脱离不了重组,所以重组本身并不是大问题,我们需要注意的是尽量在可重组项中尽量少操作多余的工作。所以通常我们不必在意这些, 因为Compose帮我做了很多优化的工作了。 当然如果在某个UI上,有其中一部份UI变化相对比较频繁,而其他则很少,此时我建议分割出来

原文

相关推荐
TDengine (老段)1 天前
TDengine 字符串函数 POSITION 用户手册
android·java·大数据·数据库·物联网·时序数据库·tdengine
2501_937154931 天前
神马影视 8.8 源码 2025 版,HDR + 杜比音效 + 零卡顿
android·源码·源代码管理·机顶盒
asjhan1 天前
Android framework强制修改系统属性
android
雨白1 天前
Jetpack Compose Navigation 2.x 详解
android·android jetpack
catchadmin1 天前
PHP 依赖管理器 Composer 2.9 发布
开发语言·php·composer
Android系统攻城狮1 天前
Android内核进阶之获取DMA地址snd_pcm_sgbuf_get_addr:用法实例(九十一)
android·pcm·android内核·音频进阶·pcm硬件参数
清空mega1 天前
Android Studio移动应用基础教程(前言)
android·ide·android studio
2501_937145411 天前
2025IPTV 源码优化版实测:双架构兼容 + 可视化运维
android·源码·源代码管理·机顶盒
zhoutanooi2 天前
安卓bp文件编译学习
android·学习
aramae2 天前
MySQL数据库入门指南
android·数据库·经验分享·笔记·mysql