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

相关推荐
ECT-OS-JiuHuaShan1 小时前
功夫不负匠心人,渡劫代谢舞沧桑
android·开发语言·人工智能·算法·机器学习·kotlin·拓扑学
ZC跨境爬虫2 小时前
移动端爬虫工具Fiddler完整配置流程:PC+安卓模拟器全覆盖,零基础一次配置成功
android·前端·爬虫·测试工具·fiddler
巴德鸟3 小时前
DaVinci 常用技巧 关键帧 自动字幕 追踪 音频 冻结帧 快捷键 多轨道字幕 扩充边缘
android·编辑器·音视频·视频·davinci·davin
学习使我健康3 小时前
Android 广播介绍详情
android·开发语言·kotlin
dalancon4 小时前
AudioTrack Start 执行流程分析
android
众少成多积小致巨4 小时前
Android 初始化语言入门
android·linux·c++
Carson带你学Android4 小时前
谁才是地表最强 Android Agent 大模型?Google官方测评来了!
android·openai
followYouself5 小时前
ASM开源库实现函数耗时插桩
android·asm·asm插桩·字节码插桩
TO_ZRG5 小时前
Android Content Provider 基础
android·jvm·oracle
studyForMokey5 小时前
【Android面试】数据库
android·数据库·面试