【发布时间】:2019-12-11 00:20:47
【问题描述】:
我只是从 Kotlin 协程开始。我正在尝试使用协程轮询服务器,并希望在 Activity 或 Fragment 暂停时停止轮询并相应地恢复轮询。所以我的pollScope 的生命周期比ViewModel.viewModelScope 提供的生命周期短。我对目前的实现并不完全满意,有几个问题:
- 这是创建
pollScope的正确方法吗?我希望它在viewModelScope也被取消时取消,这就是我指定父作业的原因。 - 如果我使用
coroutineContext.cancel()取消pollJobs,为什么协程不在onResume()中启动?如果我保留一份工作清单并取消它们,它们就会开始正常工作。 - 这是整体正确的方法吗?有没有更好的方法?
import androidx.lifecycle.LifecycleOwner
import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewModelScope
import com.spruce.messenger.utils.FullLifecycleObserverAdapter
import kotlinx.coroutines.*
import java.io.IOException
import java.util.concurrent.CopyOnWriteArrayList
import kotlin.coroutines.CoroutineContext
suspend fun poll(initialDelay: Long = 5000,
maxDelay: Long = 30000,
factor: Double = 2.0,
block: suspend () -> Unit) {
var currentDelay = initialDelay
while (true) {
try {
try {
block()
currentDelay = initialDelay
} catch (e: IOException) {
currentDelay = (currentDelay * factor).toLong().coerceAtMost(maxDelay)
}
delay(currentDelay)
yield()
} catch (e: CancellationException) {
break
}
}
}
class MyDataModel : ViewModel() {
val pollScope = CloseableCoroutineScope(SupervisorJob(parent = viewModelScope.coroutineContext[Job]) + Dispatchers.Main)
private val pollJobs = CopyOnWriteArrayList<Job>()
inner class CloseableCoroutineScope(context: CoroutineContext) : FullLifecycleObserverAdapter(), CoroutineScope {
override val coroutineContext: CoroutineContext = context
override fun onPause(owner: LifecycleOwner) {
super.onPause(owner)
// coroutineContext.cancel() cancels it but then coroutine doesn't start again in onResume() thats why cancelling jobs instead
pollJobs.forEach { it.cancel() }
}
override fun onResume(owner: LifecycleOwner) {
super.onResume(owner)
refresh()
}
}
fun refresh() {
if (pollJobs.count { it.isActive } == 0) {
startPoll()
}
}
private fun startPoll() = pollScope.launch {
try {
poll {
//fetch data from server
}
} catch (e: Exception) {
//ignore
}
}.also {
track(it)
}
private fun track(job: Job) {
pollJobs.add(job)
job.invokeOnCompletion {
pollJobs.remove(job)
}
}
}
然后在我的片段中添加 pollScope 作为生命周期观察者viewLifecycleOwner.lifecycle.addObserver(viewModel.pollScope)。
【问题讨论】:
-
我不认为 Kotlin 协程是你要找的
-
我认为 Kotlin 协程是他绝对可以使用的东西。
标签: android kotlin coroutine kotlin-coroutines