Android Jetpack Compose基础之State状态

Android Jetpack Compose基础之State状态

Stateless和Stateful

什么是Stateless,让我看下面代码

kotlin 复制代码
@Composable
fun Greeting(name:String){
    Text(text = "Hello $name")
}

Greeting方法中未持有或访问了某些状态,它唯一变化的来自参数,所以他就是一个无状态的纯函数;

Stateless它有什么特点呢

1、当无状态函数的参数没有发生改变时,不会参与调用方的重组,将重组范围局限在无状态函数外部;

Stateless它是如何实现避免无效重组的呢

Compose编译器在@Composable注解的函数体内进行了字节码插桩技术,在Text调用之前对参数进行了判断,如果参数没有变化,则跳过对Text的调用,从而避免无效重组。

什么是Stateful,让我看下面代码

kotlin 复制代码
@Composable
fun GreetingStateful(){
    var name by remember{
        mutableStateOf("World")
    }
    Text(text = "Stateful")
    Text(text = "Hello $name", modifier = Modifier.clickable {
        name ="World ${Random.nextInt(100)}"
    })
}

在代码内部我们创建了状态name来记录最新的值,GreetingStateful中依赖对name的读写,因此它时一个Stateful。

Compose中的State

在Compose中使用State描述一个状态,当Composable对state的值进行读取的同时会与state建立订阅关系,当值发生改变时,Composable会自动重组刷新ui

kotlin 复制代码
@Stable
interface State<out T> {
    val value: T//不可修改
}

State子类之MutableState

如果需要对Sate的value进行修改,则需要使用MutableState来表示可修改状态

kotlin 复制代码
@Stable
interface MutableState<T> : State<T> {
    override var value: T //var可修改类型
    operator fun component1(): T
    operator fun component2(): (T) -> Unit
}

创建方式1:直接赋值

kotlin 复制代码
    val name = mutableStateOf("default")

创建方式2:解构

kotlin 复制代码
	val(name2,setName2) = mutableStateOf("default")

创建方式3:属性代理

kotlin 复制代码
	var name3 by mutableStateOf("default")

状态的恢复与持久化

remember

1、未避免因重组导致状态的丢失,我们常见的是使用remember实现缓存状态

kotlin 复制代码
    var name4 by remember {
        mutableStateOf("default")
    }

rememberSaveable

2、虽然remember能跨越重组,但是不能跨越Acitivity和进程,这时我们需要使用rememberSaveable

kotlin 复制代码
    var name5 by rememberSaveable {
        mutableStateOf("default")
    }

rememberSaveable 中的数据会在onSaveInstanceState中以Bundle形式进行保存,在进程或者activity重建时根据key恢复到对应的Composable中,这个key是Compsable在编译期被确定的唯一表述,其源码如下

kotlin 复制代码
@Composable
fun <T : Any> rememberSaveable(
    vararg inputs: Any?,
    saver: Saver<T, out Any> = autoSaver(),
    key: String? = null,
    init: () -> T
): T {
    val compositeKey = currentCompositeKeyHash
    // key is the one provided by the user or the one generated by the compose runtime
    val finalKey = if (!key.isNullOrEmpty()) {
        key
    } else {
        compositeKey.toString(MaxSupportedRadix)
    }
    @Suppress("UNCHECKED_CAST")
    (saver as Saver<T, Any>)

    val registry = LocalSaveableStateRegistry.current

    val holder = remember {
        // value is restored using the registry or created via [init] lambda
        val restored = registry?.consumeRestored(finalKey)?.let {
            saver.restore(it)
        }
        val finalValue = restored ?: init()
        SaveableHolder(saver, registry, finalKey, finalValue, inputs)
    }

    val value = holder.getValueIfInputsDidntChange(inputs) ?: init()
    SideEffect {
        holder.update(saver, registry, finalKey, value, inputs)
    }

    return value
}

其它方式:

复制代码
如ViewModel等等方式,不再赘述

------note end------

原理有待探索~~

相关推荐
Libraeking4 小时前
破壁行动:在旧项目中丝滑嵌入 Compose(混合开发实战)
android·经验分享·android jetpack
市场部需要一个软件开发岗位4 小时前
JAVA开发常见安全问题:Cookie 中明文存储用户名、密码
android·java·安全
JMchen1236 小时前
Android后台服务与网络保活:WorkManager的实战应用
android·java·网络·kotlin·php·android-studio
crmscs7 小时前
剪映永久解锁版/电脑版永久会员VIP/安卓SVIP手机永久版下载
android·智能手机·电脑
localbob7 小时前
杀戮尖塔 v6 MOD整合版(Slay the Spire)安卓+PC端免安装中文版分享 卡牌肉鸽神作!杀戮尖塔中文版,电脑和手机都能玩!杀戮尖塔.exe 杀戮尖塔.apk
android·杀戮尖塔apk·杀戮尖塔exe·游戏分享
机建狂魔7 小时前
手机秒变电影机:Blackmagic Camera + LUT滤镜包的专业级视频解决方案
android·拍照·摄影·lut滤镜·拍摄·摄像·录像
hudawei9967 小时前
flutter和Android动画的对比
android·flutter·动画
lxysbly9 小时前
md模拟器安卓版带金手指2026
android
儿歌八万首9 小时前
硬核春节:用 Compose 打造“赛博鞭炮”
android·kotlin·compose·春节
消失的旧时光-194312 小时前
从 Kotlin 到 Dart:为什么 sealed 是处理「多种返回结果」的最佳方式?
android·开发语言·flutter·架构·kotlin·sealed