Android Compose的重组中的一段话的理解

切勿依赖于执行可组合函数所产生的附带效应,因为可能会跳过函数的重组。如果您这样做,用户可能会在您的应用中遇到奇怪且不可预测的行为。附带效应是指对应用的其余部分可见的任何更改。例如,以下操作全部都是危险的附带效应:

  • 写入共享对象的属性
  • 更新 ViewModel 中的可观察项
  • 更新共享偏好设置

一开始接触compose的时候,对这句话也是不理解的状态,现在回过头来看,感觉有点能理解了

以官方的一个例子为例:

 

//实例化SharePreference对象
val mainSharePref: SharedPreferences = MyApplication.getInstance().getSharedPreferences("main_share", android.content.Context.MODE_PRIVATE)
val value: Boolean = mainSharePref.getBoolean("isSelected",false)
val editor: SharedPreferences.Editor = mainSharePref.edit()
@Composable
fun SharedPreDemo(){
    Column() {
        SharedPrefsToggle("是否选中",value){
            editor.putBoolean("isSelected",value)
            editor.commit()
        }
    }
}
@Composable
fun SharedPrefsToggle(
    text: String,
    value: Boolean,
    onValueChanged: (Boolean) -> Unit
) {
    Row {
        Text(text)
        Checkbox(checked = value, onCheckedChange = onValueChanged)
    }
}

以上代码会创建一个可组合项以更新 SharedPreferences 中的值。该可组合项不应从共享偏好设置本身读取或写入。

运行上面的代码,会发现点击CheckBox时,会发现一个奇怪且不可预测的行为(其实就是不生效)

那么应该怎么写呢

官方方法:于是此代码将读取和写入操作移至后台协程中的 ViewModel。应用逻辑会使用回调传递当前值以触发更新。(似懂非懂)

改了一下代码,感觉应该是这个样子

@Composable
fun SharedPrefsToggle1(
    text: String,
    value: MutableState<Boolean>,
) {
    Row {
        Text(text)
        Checkbox(checked = value.value, onCheckedChange = {
            value.value=it
            GlobalScope.launch {
                editor.putBoolean("isSelected",value.value)
                editor.commit()
            }
        })
    }
}

在checkbox里的onCheckedChange回调方法里直接使用kotlin协程来进行更新。(不知道协程是什么?建议百度一下,大概意思就是功能和线程差不多,但比线程更轻量)

现在来看这句话:

切勿依赖于执行可组合函数所产生的附带效应,因为可能会跳过函数的重组

这个附带效应应该指的是回调的这些东西,就像第一个例子,无论怎么点击checkbox,都没有效果,相当于跳过了函数的重组

附上整个demo

//实例化SharePreference对象
val mainSharePref: SharedPreferences = MyApplication.getInstance().getSharedPreferences("main_share", android.content.Context.MODE_PRIVATE)
val value: Boolean = mainSharePref.getBoolean("isSelected",false)
val editor: SharedPreferences.Editor = mainSharePref.edit()
@Composable
fun SharedPreDemo(){
    Column() {
        SharedPrefsToggle("是否选中",value){
            editor.putBoolean("isSelected",value)
            editor.commit()
        }

        val value1= remember {
            mutableStateOf(value)
        }
        SharedPrefsToggle1("是否选中",value1)
    }
}

@Composable
fun SharedPrefsToggle(
    text: String,
    value: Boolean,
    onValueChanged: (Boolean) -> Unit
) {
    Row {
        Text(text)
        Checkbox(checked = value, onCheckedChange = onValueChanged)
    }
}

@Composable
fun SharedPrefsToggle1(
    text: String,
    value: MutableState<Boolean>,
) {
    Row {
        Text(text)
        Checkbox(checked = value.value, onCheckedChange = {
            value.value=it
            GlobalScope.launch {
                editor.putBoolean("isSelected",value.value)
                editor.commit()
            }
        })
    }
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

淘气章鱼哥

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值