admin管理员组文章数量:1345058
Here my composable
@Composable
fun CounterScreen(viewModel: MyViewModel = viewModel()) {
val uiState by viewModel.uiState.collectAsState()
Column(
modifier = Modifier.fillMaxSize(),
verticalArrangement = Arrangement.Center,
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(text = "Counter: ${uiState.count}")
Spacer(modifier = Modifier.height(16.dp))
FloatingActionButton(
onClick = {
viewModel.increment()
},
) {
Icon(imageVector = Icons.Default.Add, contentDescription = "Increment")
}
}
}
Why the state is preserved with recomposition when I wrote fun CounterScreen(viewModel: MyViewModel = viewModel())
but not with fun CounterScreen(viewModel: MyViewModel = MyViewModel())
. However they both create a new instance of MyViewModel.
Does someone can explain me? Thanks
Here my composable
@Composable
fun CounterScreen(viewModel: MyViewModel = viewModel()) {
val uiState by viewModel.uiState.collectAsState()
Column(
modifier = Modifier.fillMaxSize(),
verticalArrangement = Arrangement.Center,
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(text = "Counter: ${uiState.count}")
Spacer(modifier = Modifier.height(16.dp))
FloatingActionButton(
onClick = {
viewModel.increment()
},
) {
Icon(imageVector = Icons.Default.Add, contentDescription = "Increment")
}
}
}
Why the state is preserved with recomposition when I wrote fun CounterScreen(viewModel: MyViewModel = viewModel())
but not with fun CounterScreen(viewModel: MyViewModel = MyViewModel())
. However they both create a new instance of MyViewModel.
Does someone can explain me? Thanks
2 Answers
Reset to default 1With MyViewModel()
you always create a new instance.
The viewModel()
function, on the other hand, uses a view model factory behind the scenes to obtain an instance. And that factory provides the same instance for the same ViewModelStoreOwner. The latter defaults to LocalViewModelStoreOwner.current
.
Until that changes, you will always get the same instance. From the documentation:
Returns an existing
ViewModel
or creates a new one in the given owner (usually, a fragment or an activity), defaulting to the owner provided byLocalViewModelStoreOwner
.The created
ViewModel
is associated with the givenviewModelStoreOwner
and will be retained as long as the owner is alive (e. g. if it is an activity, until it is finished or process is killed).
Please have a look at the official documentation for the viewModel()
function:
viewModel()
returns an existingViewModel
or creates a new one. By default, the returnedViewModel
is scoped to the enclosing activity, fragment or navigation destination, and is retained as long as the scope is alive.
So when you use viewModel()
in any Composable in the same Activity
/ Fragment
/ NavGraph
destination, it will only create a new instance at the first time, and all other invocations will return the existing ViewModel instance.
You should be able to see the difference once you extend your code sample as follows:
@Composable
fun CounterScreens() {
Column {
CounterScreenA()
CounterScreenB()
}
}
@Composable
fun CounterScreenA(viewModel: MyViewModel = viewModel()) {
val uiState by viewModel.uiState.collectAsState()
Column(
modifier = Modifier.fillMaxWidth(),
verticalArrangement = Arrangement.Center,
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(text = "Counter: ${uiState.count}")
Spacer(modifier = Modifier.height(16.dp))
FloatingActionButton(
onClick = {
viewModel.increment()
},
) {
Icon(imageVector = Icons.Default.Add, contentDescription = "Increment")
}
}
}
@Composable
fun CounterScreenB(viewModel: MyViewModel = viewModel()) {
val uiState by viewModel.uiState.collectAsState()
Column(
modifier = Modifier.fillMaxWidth(),
verticalArrangement = Arrangement.Center,
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(text = "Counter: ${uiState.count}")
Spacer(modifier = Modifier.height(16.dp))
FloatingActionButton(
onClick = {
viewModel.increment()
},
) {
Icon(imageVector = Icons.Default.Add, contentDescription = "Increment")
}
}
}
CounterScreenA
and CounterScreenB
will now share the same ViewModel instance and will use the same counter. If you replace viewModel()
with MyViewModel()
, each of them will have their own counter.
本文标签:
版权声明:本文标题:What's the difference between "viewModel: MyViewModel = viewModel()" and "viewModel: MyViewMo 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743788572a2539140.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论