文章目录
- [Compose 怎么更新界面?](#Compose 怎么更新界面?)
-
- [Compose 怎么做到自动更新?](#Compose 怎么做到自动更新?)
- [by 委托便捷写法](#by 委托便捷写法)
- [用 State 什么时候用 by,什么时候用 =?](#用 State 什么时候用 by,什么时候用 =?)
- 总结
Compose 怎么更新界面?
Compose 既然是声明式 UI,那肯定就不能像传统 UI 那样命令式的方式更新界面,声明式的界面更新就是,你把值改了界面就会自动更新,你的参数会被 Compose 自动监听。
java
class MainActivity : ComponentActivity() {
// 注意这里是用 =
// mutableStateOf 返回的 MutableState 对象
val name = mutableStateOf("name")
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
Box(Modifier.safeContentPadding()) {
// 因为 mutableStateOf 用的 =,赋值使用 name.value
Text(name.value)
}
}
lifecycleScope.launch {
delay(3000)
// 修改数值,就能通知到 Composable 刷新
name.value = "vincent"
}
}
}
上面是一个文字的 Composable,开启了一个协程 3s 后修改 name 的值,Compose 监听到 name 改变了,就会同步将 Text() 也更新。
需要注意的是,在 Compose 中要实现参数被监听,参数要使用 Compose 提供的 State 接口,把值装到它的里面才能被监听,真正被订阅的是这个 State 对象:
java
SnahshotState.kt
@Stable
public interface State<out T> {
public val value: T
}
State 还有个子接口 MutableState:
java
SnahshotState.kt
@Stable
public interface MutableState<T> : State<T> {
override var value: T
public operator fun component1(): T
public operator fun component2(): (T) -> Unit
}
MutableState 在写法上使用 mutableStateOf 函数创建:
java
val name = mutableStateOf("name")
Compose 怎么做到自动更新?
从最直观的角度解释,对于一段程序要执行更新,那么就是要重新执行这段代码,从上面的例子说就是 Text(name.value) 被重新执行了。
但是在运行时是不能指定执行某一段代码的,能做到的是找到用到这个变量的地方,拿到所在的函数类型的对象重新调用整个代码块,这也是 Compose 能做到自动更新的简单原理:
java
// 函数类型对象 content
Box(Modifier.safeContentPadding(), content = {
Text(name.value)
})
by 委托便捷写法
每次使用 MutableState 都要 .value 才能使用很麻烦,Compose 为我们提供了简洁处理,帮我们去掉了这个操作,通过 by 将 setValue 和 getValue 都交由 mutableStateOf 委托处理。
java
// name 实际类型此时是 String,mutableStateOf 通过 by 变成了代理类型
// val name = mutableStateOf("name")
var name by mutableStateOf("name")
setContent {
// Text(name.value)
Text(name)
}
lifecycleScope.launch {
delay(3000)
// name.value = "vincent"
name = "vincent"
}
Compose 引入了一个扩展函数 androidx.compose.runtime.setValue 和 androidx.compose.runtime.getValue,将 MutableState 类型转换为调用的数值类型,但目前仍需要手动导入这个扩展。
用 State 什么时候用 by,什么时候用 =?
我们先看下面的例子:
java
class MainActivity : ComponentActivity() {
var name by mutableStateOf("name")
var name1 = name
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
Box(Modifier.safeContentPadding()) {
// 注意这里是 name1
Text(name1)
}
}
lifecycleScope.launch {
delay(3000)
// 注意这里修改的是 name
name = "vincent"
}
}
}
运行上面的代码,会发现 3s 后代码不会自动更新,因为 Text(name1) 传的是 name1,修改的是 name,修改 name 时找不到使用的 Composable,所以不会更新。简单原理上面已经讲过,因为监听的对象是 State,name 使用的 by 声明,其实它已经是一个 String 类型,而不是 State。
要上面的代码生效,就要改为用 =:
java
class MainActivity : ComponentActivity() {
var name = mutableStateOf("name") // 改成了 =
var name1 = name // name 赋值给 name1 此时才是 State 对象
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
Box(Modifier.safeContentPadding()) {
// 注意这里是 name1
Text(name1.value)
}
}
lifecycleScope.launch {
delay(3000)
// 注意这里修改的是 name
name.value = "vincent"
}
}
}
by 的代理功能虽然在日常使用时很方便,但它的代理功能是不能通过赋值来传递的,也会丢失订阅监听,这种间接使用的场景就需要用 =。
总结
-
Compose 的自动更新是通过 State 对象来实现的,使用 State 才能被订阅监听,在日常使用中是使用它的子接口 MutableState,通过 mutableStateOf 创建
-
使用 by 关键字委托能更方便的使用 MutableState,但是在一些间接场景还是需要用 =