【发布时间】:2020-01-23 15:02:52
【问题描述】:
我知道建议是在我们的 Activity 中使用 ViewModel,因此我们可以使用它的viewModelScope。由于 ViewModel 比 Activity 寿命更长,我们不必取消 activity.onDestroy() 中的作业。
但是,有时您的 Activity 非常简单。例如,它可以使用已安装的过滤包填充列表视图。您可以非常简单地使用委托为活动创建范围,并取消 onDestroy() 中的作业:
class MyActivity(): AppCompatActivity(), CoroutineScope by MainScope() {
private val listAdapter = MyAdapter()
override fun onCreate() {
super.onCreate()
setContentView(R.layout.my_activity)
recycler_view.apply {
layoutManager = LinearLayoutManager(this)
adapter = listAdapter
}
launch {
val packages = getOrgPackagesWithIcons()
adapter.apply {
data = packages
notifyDataSetChanged()
}
}
}
override fun onDestroy() {
super.onDestroy()
cancel() // CoroutineContext
}
private suspend fun getOrgPackagesWithIcons() = withContext(Dispatchers.Default) {
var toNextYield = 20
packageManager.getInstalledPackages(0)
.filter { it.packageName.startsWith("org")
.take(100)
.map {
if (--toNextYield == 0) { // Make it cancellable
toNextYield = 20
yield()
}
MyPackageData(
it.applicationInfo.loadLabel(packageManager).toString(),
it.packageName,
it.applicationInfo.loadIcon(packageManager)
)
}
}
}
对于这种情况,ViewModel 感觉有点矫枉过正。它只是抽象 PackageManager 的另一层,它本身就是一个视图模型。
上面的代码可以很容易地在后台组装数据。问题是当屏幕旋转时,或在其他配置更改期间,协程被取消并重新启动。有没有一个干净的方法可以通过像这样一个非常简单的 Activity 的配置更改来保持 CoroutineScope 活着?
onRetainNonConfigurationInstance() 已弃用。我想我们可以把它放在一个 Fragment 中并使用retainInstance = true,但是在这样一个简单的 Activity 中引入一个 Fragment 层也感觉有点矫枉过正。
也许有一种方法可以创建一个空的 ViewModel 实现,以便我们可以借用它的范围?
【问题讨论】:
标签: android kotlin kotlin-coroutines