我的viewModel里有两个州流。为了收集碎片,我必须发射两次协同线,如下所示:
lifecycleScope.launchWhenStarted {
stocksVM.quotes.collect {
if (it is Resource.Success) {
it.data?.let { list ->
quoteAdapter.submitData(list)
}
}
}
}
lifecycleScope.launchWhenStarted {
stocksVM.stockUpdate.collect {
log(it.data?.data.toString())
}
}如果我有更多的州流,我必须分别发射协同。在我的片段/活动中是否有更好的方法来处理多状态流?
发布于 2021-06-02 07:55:20
您将需要不同的协同,因为collect()是一个挂起函数,它一直挂起直到Flow终止。
对于收集多个流,目前推荐的方法是:
lifecycleScope.launch {
lifecycle.repeatOnLifecycle(Lifecycle.State.STARTED) {
launch {
stocksVM.quotes.collect { ... }
}
launch {
stocksVM.stockUpdate.collect { ... }
}
}
}请注意,launchWhenStarted的问题是,虽然您的新发出的项目不会被处理,但您的生产者仍将在后台运行。
我肯定会读一读,因为它很好地解释了当前的最佳实践:https://medium.com/androiddevelopers/a-safer-way-to-collect-flows-from-android-uis-23080b1f8bda。
发布于 2021-06-02 06:25:49
您可以选择混合多个流。
在merge或combine中使用kotlin函数。当然,这两个函数的用法是不同的。
添加:
如果流未被处理,则打开多个Coroutines以收集():
fun main() {
collectFlow()
}
fun emitStringElem(): Flow<String> = flow {
repeat(5) {
delay(10)
emit("elem_$it")
}
}
fun emitIntElem(): Flow<Int> = flow {
repeat(10) {
delay(10)
emit(it)
}
}打开两个协同收集结果是:
From int Flow: item is: 0
From string Flow: item is: elem_0
From int Flow: item is: 1
From string Flow: item is: elem_1
From int Flow: item is: 2
From string Flow: item is: elem_2
From int Flow: item is: 3
From string Flow: item is: elem_3
From int Flow: item is: 4
From string Flow: item is: elem_4
From int Flow: item is: 5
From int Flow: item is: 6
From int Flow: item is: 7
From int Flow: item is: 8
From int Flow: item is: 9合并两个流
fun margeFlow() = runBlocking {
merge(
emitIntElem().map {
it.toString()
}, emitStringElem()
).collect {
println(it)
}
}结果是:
0
elem_0
1
elem_1
2
elem_2
3
elem_3
4
elem_4
5
6
7
8
9两种流动:
fun combineFlow() = runBlocking {
combine(emitIntElem(), emitStringElem()) { int: Int, str: String ->
"$int combine $str"
}.collect {
println(it)
}
}结果是:
0 combine elem_0
1 combine elem_0
1 combine elem_1
2 combine elem_2
3 combine elem_3
4 combine elem_4
5 combine elem_4
6 combine elem_4
7 combine elem_4
8 combine elem_4
9 combine elem_4发布于 2021-06-02 13:42:32
正如@RóbertNagy所说,你不应该使用launchWhenStarted。但是,在不需要执行嵌套的launches的情况下,有一种替代的语法可以用正确的方式完成它:
stocksVM.quotes
.flowOnLifecycle(Lifecycle.State.STARTED)
.onEach {
if (it is Resource.Success) {
it.data?.let { list ->
quoteAdapter.submitData(list)
}
}
}.launchIn(lifecycleScope)
stocksVM.stockUpdate
.flowOnLifecycle(Lifecycle.State.STARTED)
.onEach {
log(it.data?.data.toString())
}.launchIn(lifecycleScope)https://stackoverflow.com/questions/67799859
复制相似问题