为什么在每个 Compose Navigation 路线中都会创建一个新的 ViewModel?

jpm*_*jpm 20 android kotlin android-jetpack-compose

我有一个活动应用程序,仅使用 ui 的可组合项(一个活动,没有片段)。我使用一个视图模型在两个不同的屏幕(可组合项)中保存 ui 的数据。我按照状态文档中的描述在两个屏幕中创建视图模型

@Composable
fun HelloScreen(helloViewModel: HelloViewModel = viewModel()) 
Run Code Online (Sandbox Code Playgroud)

现在我注意到在第一个屏幕中加载或设置的数据在第二个屏幕中被重置。

我还注意到init{}每次被调用时viewModel()都会被调用。这真的是预期的行为吗?

根据该方法的文档,它应该返回现有的 ViewModel 或创建一个新的 ViewModel。

我还看到视图模型是不同的对象。所以viewModel()总是创建一个新的。但为什么?

有什么想法我可能做错了什么吗?或者我误解了该方法的用法?

Phi*_*hov 31

通常,视图模型在整个可组合项范围内共享,并且init不应被多次调用。

但如果您使用撰写导航,它会为每个目的地创建一个新的模型商店所有者。如果您需要在目的地之间共享模型,可以通过两种方式进行:

  1. 通过直接传递它来viewModel调用。在这种情况下,只有传递的视图模型将绑定到父商店所有者,并且内部创建的所有其他视图模型将绑定(因此当从堆栈中删除路由时销毁)到当前路由。
  2. 通过证明 的值LocalViewModelStoreOwner,因此内部的所有可组合项都将绑定到父视图模型存储所有者,因此当从堆栈中删除路由时不会被释放。
val viewModelStoreOwner = checkNotNull(LocalViewModelStoreOwner.current) {
    "No ViewModelStoreOwner was provided via LocalViewModelStoreOwner"
}
val navController = rememberNavController()
NavHost(navController = navController, startDestination = "first") {
    composable("first") {
        val model = viewModel<Model>(viewModelStoreOwner = viewModelStoreOwner)
    }
    composable("second") {
        CompositionLocalProvider(
            LocalViewModelStoreOwner provides viewModelStoreOwner
        ) {
            val model = viewModel<Model>()
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

  • Hej Philip 感谢您向我介绍 CompositionLocal!我使用了第二种方法,遇到了一些困难,但现在可以了!如果其他人偶然发现了这一点,这对我有帮助:https://developer.android.com/reference/kotlin/androidx/compose/runtime/CompositionLocal 和 https://developer.android.com/jetpack/compose/compositionlocal (2认同)