Kotlin等待与协程异步

时间:2020-10-21 17:59:32

标签: android kotlin kotlin-coroutines

我想在实例化phoneViewModel和ScanViewModel时打开一个新活动。通过调用异步函数InitialRead()实例化它们。我正在记录每个步骤,atm将它们记录为done3 => done2 => done1

我想按以下顺序排列它们: done1 => done2 => done3

我有以下代码:

class MainBusinessActivity : AppCompatActivity() {

private lateinit var scanViewModel: ScanViewModel
private lateinit var phoneViewModel: PhoneViewModel

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_main_business)
}

private fun startEntitySetListActivity() = GlobalScope.async {
    val sapServiceManager = (application as SAPWizardApplication).sapServiceManager
    sapServiceManager?.openODataStore {
        phoneViewModel =  ViewModelProvider(this@MainBusinessActivity).get(PhoneViewModel::class.java).also {it.initialRead{Log.e("done", "done1")}}
        scanViewModel = ViewModelProvider(this@MainBusinessActivity).get(ScanViewModel::class.java).also {it.initialRead{Log.e("done", "done2")}}
    }
}

override fun onResume() {
    super.onResume()
    //startEntitySetListActivity()
    runBlocking {
        startEntitySetListActivity().await()
        val intent = Intent(this@MainBusinessActivity, HomeActivity::class.java)
        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK)
        Log.e("done", "done3")
        startActivity(intent)
    }
}

}

我在做什么错?有人可以纠正我的代码吗?

1 个答案:

答案 0 :(得分:2)

请勿在Android应用中使用runBlockingrunBlocking完全违背了使用协程的目的,并可能导致ANR。您也可能永远不要使用GlobalScope,这会导致UI泄漏。对于某些长时间运行的任务,您可能需要它,而这种任务没有必要放入服务但不依赖于任何UI组件,但我想不出任何示例

您也不应该在后台实例化ViewModel。该操作应在onCreate()中完成。

将此函数设为暂停函数,它可以在返回之前同时在后台分解两个任务。

lifecycleScope开始协程。

假设sapServiceManager?.openODataStore是一个需要回调的异步任务,则需要将其包装在suspendCoroutine中。

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_main_business)
    phoneViewModel = ViewModelProvider(this@MainBusinessActivity).get(PhoneViewModel::class.java)
    scanViewModel = ViewModelProvider(this@MainBusinessActivity).get(ScanViewModel::class.java)
}

private suspend fun startEntitySetListActivity() = coroutineScope {
    val sapServiceManager = (application as SAPWizardApplication).sapServiceManager
    sapServiceManager ?: return
    suspendCoroutine<Unit> { continuation ->
        sapServiceManager.openODataStore { continuation.resume(Unit) }
    }
    listOf(
        launch {
            phoneViewModel.initialRead{Log.e("done", "done1")}
        },
        launch {
            scanViewModel.initialRead{Log.e("done", "done2")}
        }
    ).joinAll()
}

override fun onResume() {
    super.onResume()
    lifecycleScope.launch {
        startEntitySetListActivity()
        val intent = Intent(this@MainBusinessActivity, HomeActivity::class.java)
        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK)
        Log.e("done", "done3")
        startActivity(intent)
    }
}