△ Jetsnack 应用屏幕截图
△ 实践 | Jetpack Compose 中的状态管理
Bilibili 视频链接
https://www.bilibili.com/video/BV1t5411m7rh/
在 Compose 中使用 State
Jetsnack 是一款使用 Compose 构建的小吃订购示例应用,状态对它来说非常重要。比如,在一屏中显示哪些商品、显示用户筛选的小吃以及记录购物车等操作,都需要状态的支持。我们将 Compose 构建的界面称之为组合 (Composition),它会在屏幕中呈现应用的当前状态。下图直观地展示了组合在视觉上呈现搜索页的过程,您可以在其中找到搜索栏 (SearchBar)、分隔线 (JetsnackDivider) 和搜索建议 (SearchSuggestions),这些都是搜索界面的组成部分:
△ 组合呈现搜索界面的过程
在像 Compose 这样的声明式框架中,您只需描述应用的当前状态,Compose 会负责在状态发生更改时更新界面。因此,当我们导航到购物车屏幕时,Compose 也会重新执行受状态更改影响的部分界面。下图中,NavHost 更新为显示购物车界面。由于界面的每个部分都是一个可组合项,当状态更改时,这些函数会进行重组,以便在屏幕上显示新数据:
△ 组合呈现购物车界面的过程
在购物车界面中,我们重点关注单独的购物车商品项。该元素用于显示购物车中的商品,并允许您更改数量:
△ 单独的购物车商品
我们可以使用包含两个 Button 和一个 Text 的 Row 来构建该界面,但是要如何记录购物车中商品的当前数量呢?
fun CartItem() {var quantity = ... ¯\_(ッ)_/¯ ... ?Row {Button(onClick = { quantity++ }) {Text("+")}Text(quantity.toString())Button(onClick = { quantity-- }) {Text("-")}}}
fun CartItem() {var quantity: Int = 1...}
Compose 具有特殊的状态跟踪系统,可以在某个状态改变时,重组读取该状态的所有可组合项。这种机制使得 Compose 可以对界面进行精细控制,在状态发生改变时不用修改整个界面,只需重组需要更改的可组合函数即可。
这一功能是通过跟踪状态写入 (即状态更改) 以及状态读取来实现的,我们可以使用 Compose 的 State 和 MutableState 类型使状态可被观察。Compose 会跟踪读取 State 中 value 属性的可组合项,并在 value 发生更改时触发重新组合。
// State<T>interface State<out T> {val value: T}// MutableState<T>interface MutableState<T> : State<T> {override var value: T}
您可以使用 mutableStateOf 函数创建 MutableState,该函数需要接收一个初始值,并且它的 value 是可变的。相应的,我们需要改用 value 属性来读取和写入 quantity 状态:
fun CartItem() {val quantity: MutableState<Int> = mutableStateOf(1)Row {Button(onClick = { quantity.value++ }) {Text("+")}Text(quantity.value.toString())Button(onClick = { quantity.value-- }) {Text("-")}}}
但是,即使 Compose 已经跟踪了 quantity 变量,并触发了重组,您会发现界面依然没有显示状态的更改。问题在于,虽然该函数已经重组,但 quantity 的值 value 总是会被初始化为 1。这是一个常见的错误,因此您在尝试编写这段代码时也会产生编译错误。为了在重组中重用 quantity 状态,我们需要使其成为组合的一部分。要做到这一点,可以使用 remember 可组合函数将对象存储在组合中:
fun CartItem() {val quantity = remember { mutableStateOf(1) }...}
Remember 可用于存储可变对象和不可变对象,您必须对在组合中创建的 State,也就是可组合函数中的 State 执行 remember 操作。在被记住后,状态将成为组合的一部分,并在函数重组时被重用,这样一来,购物车商品也可以按照我们的预期工作了。
由于重新组合期间会保留 quantity,因此屏幕上将显示改变后的新值。此外,Compose 还提供了 rememberSaveable,其行为与 remember 类似,但存储的值可在 Activity 和进程重建后保留下来,这是在配置变更时保留界面数据的好方法。rememberSaveable 适用于界面状态,如商品数量或选定的标签,但不适用诸如过渡动画状态一类的用例。
此外,您还可以将委托属性与 State API 结合使用。在下面的代码中可以看到,在实际应用中,我们可以使用 by 关键字来实现这一点。如果您不想每次都访问 value 属性的话,这不失为一种好方法。
fun CartItem() {var quantity: Int by rememberSaveable { mutableStateOf(1) }Row {Button(onClick = { quantity++ }) {Text("+")}Text(quantity.toString())Button(onClick = { quantity-- }) {Text("-")}}}
注意,您只应在可组合函数的作用域之外操作状态,因为可组合项可能会频繁地、以任何顺序执行。上面的代码中,在 onClick 监听器中修改 quantity 的操作是安全的,因为 onClick 不是可组合函数。您可以根据特定的用户输入触发状态更改,例如点击按钮或使用附带效应:
状态提升
为了使 CartItem 可被重用,我们将 quantity 状态从 CartItem 提升至 Cart 中,这一过程被称为状态提升:
状态提升是一种将私有状态移出可组合项的模式,这可以使可组合项更趋于无状态,从而提高在应用中的可重用性。无状态可组合项是指不保存任何私有状态的可组合项。理想情况下,可组合项应接收状态作为参数,并使用 lambda 向上传递事件:
△ 可组合项应接收状态 (State)
使可组合项趋于无状态,不但可以使其符合单一可信来源原则,而且可以提高它的可重用性和可测试性。因为在这种情况下,可组合项没有与任何特定的数据处理方式耦合在一起,而我们还可以共享和拦截以这种方式提升的状态。下面是无状态版的 CartItem 的示例代码,它接收 quantity 并做为状态显示,同时将用户交互公开为事件:
fun CartItem(quantity: Int, // 状态incrementQuantity: () -> Unit, // 事件decrementQuantity: () -> Unit // 事件) {Row {Button(onClick = incrementQuantity) {Text("+")}Text(quantity.toString())Button(onClick = decrementQuantity) {Text("-")}}}
接下来我们来看 Cart 可组合函数的实现。Cart 界面会在 LazyColumn 中显示不同的 CartItem,同时负责使用正确的信息调用 CartItem。Cart 中的项目实际上是从 CartViewModel 取得的应用数据。我们对于每个 CartItem 都传入特定的 quantity,增加或减少数量的逻辑被委托给 ViewModel,ViewModel 则作为 Cart 数据的持有者:
fun Cart(viewModel: CartViewModel = viewModel()) {val cartItems by viewModel.cartItemsLazyColumn {items(cartItems) { item ->CartItem(quantity = item.quantity,incrementQuantity = {viewModel.inrementQuantity(item)},decrementQuantity = {viewModel.decrementQuantity(item)})}}}
状态提升是一种在 Compose 中广泛使用的模式。作为一种拦截和控制界面元素内部使用状态的方式,您可以在大多数 Compose API 中看到它。我们也可以将拦截状态设计为可选操作,从而可以利用强大的默认参数特性。以下面的代码为例,如果需要控制或共享 scaffoldState,您可以传入该状态;而如果您没有传入,该函数也会创建一个默认状态:
fun Scaffold(scaffoldState: ScaffoldState = rememberScaffoldState(),...) { ... }public fun NavHost(navController: NavHostController,...) { ... }
fun Cart(orderLines: List<OrderLine>,removeSnack: (Long) -> Unit,increaseItemCount: (Long) -> Unit,decreaseItemCount: (Long) -> Unit,inspiredByCart: SnackCollection,modifier: Modifier = Modifier) {...}
这样的 Cart 可组合项更易于预览和测试,同时符合单一可信来源原则。这样做的可重用性也更高,比如,如果我们需要,就可以在窗口尺寸足够大的情况下,与另一个界面并排显示购物车。不仅如此,我们还提供了有状态版本,使其也可以通过特定的方式处理状态和事件:
fun Cart(modifier: Modifier = Modifier,viewModel: CartViewMo = viewModel()) {val orderLines by viewModel.orderLines.collectAsState()Cart(orderLines = orderLines,removeSnack = viewModel::removeSnack,increaseItemCount = viewModel::increaseSnackCount,decreaseItemCount = viewModel::decreaseSnackCount,inspiredByCart = viewModel.inspiredByCart,modifier = modifier)}
我们可以看到,这个版本的 Cart 通过处理业务逻辑和状态的 CartViewModel 来调用无状态版的 Cart 可组合项。这种同时提供有状态、无状态,或趋于无状态组合项的模式,可以很好的兼顾各种使用场景。您既可以在需要时重用可组合项,又可以在应用中以特定的方式使用它。
状态管理
在开始之前,我们要定义文中所涉及特定术语的含义:
fun JetsnackApp() {JetsnackTheme {val scaffoldState = rememberScaffoldState()val coroutineScope = rememberCoroutineScope()JetsnackScaffold(scaffoldState = scaffoldState) {Content(showSnackbar = { message ->coroutineScope.launch {scaffoldState.snackbarHostState.showSnackbar(message)}})}}}
我们使用 JetsnackAppState 类作为状态容器,它将会是 JetsnackApp 的界面元素状态的可信来源,因此所有状态写入都应在该类中进行。状态容器是在组合中创建和记住的普通类,因此,该类的作用域限定于创建它的可组合项。JetsnackAppState 只是一个普通类,而且由于它遵循可组合项的生命周期,因此可以使用 Compose 的依赖项,而不必担心内存泄漏:
class JetsnackAppState(// 一般的类可以接收 Compose 依赖val scaffoldState: ScaffoldState,val navController: NavHostController,...) {val shouldShowBottomBar: Boolean// 在读取的值发生改变时会进行重组get() = navController.currentBackStackEntryAsState().value?.destination?.route in bottomBarRoutes// 与界面相关的逻辑fun navigateToBottomBarRoute(route: String) {if (route != currentRoute) {navController.navigate(route) {launchSingleTop = truerestoreState = truepopUpTo(findStartDestination(navController.graph).id) {saveState = true}}}}}
状态容器还可以包含可组合项属性,更改此类属性将会触发重组,上面的代码即为是否显示底部操作栏的属性。该状态容器还包含界面相关的逻辑,比如导航逻辑。就像前面说过的,您必须使用 remember 记住数据,以便在重新组合期间重用数据,如果状态容器使用了 State 依赖项,那么应该提供方法来记住状态容器。在下面的代码中,我们将依赖项传入 remember,以便在任何依赖项发生更改时获取 JetsnackAppState 的新实例:
fun rememberJetsnackAppState(scaffoldState: ScaffoldState = rememberScaffoldState(),navController: NavHostController = rememberNavController(),...) = remember(scaffoldState, navController, ...) {JetsnackAppState(scaffoldState, navController, ...)}
现在,我们在 JetsnackApp 中获取了 appState 的新实例。我们使用该实例将被提升的状态传递给可组合项,并在需要显示界面元素时检查该状态;同时调用函数来触发与界面相关的操作:
fun JetsnackApp() {JetsnackTheme {val appState = rememberJetsnackAppState()JetsnackScaffold(scaffoldState = appState.scaffoldState,bottomBar = {if (appState.shouldShowBottomBar) {JetsnackBottomBar(tabs = appState.bottomBarTabs,navigateToRoute = {appState.navigateToBottomBarRoute(it)})}}) {NavHost(navController = appState.navController, ...) {
简单来说,状态容器是一个普通类,用于提升界面元素的状态并包含界面相关的逻辑。状态容器可以降低可组合项的复杂性,并提高其可测试性,从而有助于关注点分离。它还可以使状态提升变得更为容易,因为只需提升一个状态而不是多个状态。状态容器可以非常简单并且只用于特定用途,例如,只用于搜索界面的 SearchState 类,其中仅包含 activeFilters 和 searchResults List。当您需要跟踪状态或界面逻辑时可以使用状态容器来帮助控制复杂度。
class SearchState {var searchResults: List<Snack> by mutableStateOf(listOf())private setvar activeFilters: List<Filter> by mutableStateOf(listOf())private set...}
除了一般的状态容器外,我们还可以使用 ViewModel,这是一种继承架构组件 ViewModel 类的类。ViewModel 可用作由业务逻辑确定状态的状态容器。ViewModel 有两项责任: 首先,提供对应用业务逻辑的访问,这些业务逻辑通常位于层次结构的其他层级中,如存储区或用例中;其次,准备要在特定屏幕上呈现的应用数据,通常是用可观察类型呈现屏幕的界面状态。
在完全使用 Compose 构建的应用中,我们可以使用 Compose 的 State 类型。但在混合应用中,您还可能会用到其他的可观察类型,如 StateFlow:
class CartViewModel(private val repository: SnackRepository,private val savedState: SavedStateHandle) : ViewModel() {val uiState: State<CartUiState> = ...fun increaseSnackCount(snackId: Long) { ... }fun decreaseSnackCount(snackId: Long) { ... }fun removeSnack(snackId: Long) { ... }fun completeOrder() { ... }}
ViewModel 在配置变更后仍然有效,因此其生命周期比组合更长。ViewModel 不属于组合的一部分,因此不能接受组合作用域内的状态,比如使用记住的值,您需要谨慎对待此类操作,因为这可能会导致内存泄漏。ViewModel 依赖于层次结构的其他层级,例如存储区或用例。另外,如果您希望界面在状态发生更改时重组,您依然需要使用 Compose State API。
class CartViewModel(private val repository: SnackRepository,private val savedState: SavedStateHandle) : ViewModel() {// 在 ViewModel 中,仍要使用 State 类型来使状态可被 Compose 观察var uiState by mutableStateOf<CartUiState>(EmptyCart)private set...}
不过在本例中,由于 uiState 位于组合之外,因此您不需要记住它,而只需使用它即可。可组合函数将在 uiState 更改时重新执行:
fun Cart(viewModel: CartViewModel = viewModel()) {val uiState by viewModel.uiState}
层次结构的其他层通常使用流式数据来传播更改,您可能已经开始在 ViewModel 中使用它们了。Flow 也可以很好地与 Compose 结合使用,我们提供了工具函数,可以将数据流转换为 Compose 的可观察 State API。例如,您可以使用 collectAsState 从数据流中收集值,并将它们呈现为 State。这样一来,每当数据流发出新值时,就会触发重组。
fun Cart(viewModel: CartViewModel = viewModel()) {// 通过将 Flow 转换为 State 来跟踪 snacks 状态的改变val snacks by viewModel.snacks.collectAsState()...}
总的来说,ViewModel 可以在组合之外提升组合的状态,同时具有更长的生命周期。ViewModel 负责屏幕的业务逻辑并决定要显示哪些数据,它会从其他层级获取数据,并准备这些要呈现的数据。因此,建议在屏幕级的可组合项中使用 ViewModel。
与普通状态容器相比 ViewModel 具有一些优势,其中包括,ViewModel 触发的操作在配置变更后仍然有效,并且 ViewModel 可以与 Hilt、Navigation 等 Jetpack 库很好地集成在一起。在使用 Hilt 时,仅使用一行代码,就能通过 Hilt 提供的依赖项获取 ViewModel。
当屏幕位于返回栈中时,Navigation 会缓存 ViewModel,这意味着当返回到目标时,数据已经处于可用状态;而当目标离开返回栈后 ViewModel 又会被清除,从而确保状态可以被自动清理。
使用遵循可组合项界面生命周期的状态容器 (即使用一般的类作为状态容器),将会难以做到前述操作。尽管如此,如果 ViewModel 的优势不适用于您的用例或者您以不同的方式操作,您可以使用其他最适合您的状态容器,而不一定是 ViewModel 来完成相应的工作。
界面级可组合项也可以同时使用 ViewModel 和其他状态容器。由于 ViewModel 的生命周期更长,普通的状态容器可以将 ViewModel 作为依赖项。
我们来看一下实际应用。除了在 Cart 可组合项中使用 CartViewModel 之外,我们还可以另外使用包含界面元素状态和界面逻辑的 CartState。在 CartState 中,我们使用 lazyListState 来记录大型购物车界面的滚动位置;使用 resources 来格式化信息和价格;如果允许展开商品以显示更多信息,还可以了解每个商品的状态:
class CartState(lazyListState: LazyListState,resources: Resources,expandedItems: List<Item> = emptyList()) {...fun formatPrice(...) { ... }}
Cart 中同时使用了 ViewModel 和其他状态容器,它们具有不同的用途,并可以协同工作。我们来仔细看一下它们的生命周期: CartState 会遵循 Cart 可组合项的生命周期,当 Cart 从组合中移除后 CartState 也会一同移除;而 CartViewModel 具有不同的生命周期,即导航目的地、导航图、Fragment 或 Activity 的生命周期:
△ CartState 遵循 Cart 的生命周期
从全局来看,每个实体的作用都有明确的定义,从包含界面元素的界面层到包含业务逻辑的数据层,每个实体都有特定的用途。在下图中,您可以看到扮演着不同角色的实体,以及它们之间潜在的依赖关系:
总结
对于我们的应用来说,状态是十分重要的一部分。我们可以在 Compose 中使用 State API 做到简单的状态响应,也可以使用一般的类或者 ViewModel 作为状态容器,以便对可组合项进行状态提升,并使其符合单一可信来源原则。我们还可以组合不同的状态容器,从而利用它们各自不同的生命周期。
下面是我们在文章中列出的表格,请记住它,以便您在未来做决策时可以为您的应用提供明确的状态管理架构:
希望本文能帮助您实现 "理想的 Compose 状态",祝您拥有愉快的 Compose 使用体验。
Jetpack Compose 使用入门
https://developer.android.google.cn/jetpack/compose/documentation
https://developer.android.google.cn/codelabs/jetpack-compose-state
您可以通过下方二维码向我们提交反馈,或分享您喜欢的内容、发现的问题。您的反馈对我们非常重要,感谢您的支持!
推荐阅读