到目前为止,只要我想访问数据库,我就一直在使用这种模式:
runBlocking {
launch {
// fetch something from the database and put it to some view
}
}
现在我正在深入研究 Kotlin 协程,我越来越相信这是一个糟糕的模式。本质上,我还不如只是 allowMainThreadQueries
,因为我的模式无论如何都会阻塞主线程。
不幸的是,我还没有找到合适的模式。如何有效使用Kotlins协程访问数据库?
runBlocking
是协程的唯一入口点吗?
考虑这个场景:
override fun onCreate() {
setContentView(someLayout)
// concurrently fetch something from the database and put it in some view
// onCreate may return before this has finished
someButton.setOnClickListener {
// concurrently insert or update something in the database
}
}
最佳答案
您应该永远不要在 Android 项目中使用runBlocking
,除非您将 Kotlin 协程代码与一些不能使用协程的 Java 代码混合,并且它需要一种方法来在它自己的后台线程之一上以阻塞方式调用一些协程。在这种情况下,您可能会使用 runBlocking
创建一个桥接函数供 Java 代码调用,但您绝不会从 Kotlin 调用此函数,当然也绝不会从主线程调用它。在主线程上调用阻塞代码会卡住 UI,这会让您的应用感觉卡顿并有可能触发 ANR(应用未响应)错误。
启动协程的正确方法是使用 CoroutineScope 启动协程。 Android Jetpack 框架已经为您提供了 Activity、Fragments 和 ViewModels。
在 Activity 中,使用 lifecycleScope.launch
。在 Fragment 中,您通常应该使用 viewLifecycleOwner.lifecycleScope.launch
。在 ViewModel 中,使用 viewModelScope.launch
。
使用 CoroutineScope 而不是 runBlocking
有什么作用?它可以防止长时间运行的挂起操作(如从磁盘读取数据库)阻塞主线程并卡住您的 UI。当 Activity/Fragment/ViewModel 被拆除时,它会自动取消长时间运行的工作,因此它可以防止内存泄漏和资源浪费。
关于android - 使用协程访问数据库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72192016/