Compose:MutableState 和 mutableStateOf

文章目录

  • [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,但是在一些间接场景还是需要用 =

相关推荐
YF02112 小时前
深入剖析 Kotlin 的高效之道与核心实战
android·kotlin·app
程序员码歌3 小时前
别再让 AI 自由发挥了:OpenSpec 才是团队协作不跑偏的关键
android·前端·人工智能
敲代码的鱼3 小时前
NFC读卡能力 支持安卓/iOS/鸿蒙 UTS插件
android·ios·uni-app
刮风那天3 小时前
Android 常驻进程如何被查杀?
android
刮风那天5 小时前
Android 如何降低进程优先级可以被查杀?
android
资源分享助手6 小时前
超级改图P图改字无限制版教程(安卓)AI改图软件、图片改字软件、安卓修图APP、智能消除工具、图片拼接APP、超级改图下载
android·人工智能
Lehjy7 小时前
【Linux】文件系统磁盘存储结构
android·linux·运维
BU摆烂会噶7 小时前
【LangGraph】节点内调用与状态隔离
android·人工智能·python·ui·langchain·人机交互
BU摆烂会噶8 小时前
【LangGraph】作为节点添加与状态共享
android·人工智能·python·ui·langchain·人机交互
刘大猫.9 小时前
重塑经典:Snapseed4.0全面登陆安卓,内置“胶片相机”与专业手动模式
android·数码相机·ai·机器人·大模型·算力·snapseed4.0