android - 将广播接收器包装到流中(协程)

标签 android kotlin broadcastreceiver kotlin-coroutines android-wifi

我有一个用于 wifi 扫描结果的广播接收器作为数据源,我想以协程方式制作它。我在这里找到了暂停功能的答案:
https://stackoverflow.com/a/53520496/5938671

suspend fun getCurrentScanResult(): List<ScanResult> =
    suspendCancellableCoroutine { cont ->
        //define broadcast reciever
        val wifiScanReceiver = object : BroadcastReceiver() {
            override fun onReceive(c: Context, intent: Intent) {
                if (intent.action?.equals(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION) == true) {
                    context.unregisterReceiver(this)
                    cont.resume(wifiManager.scanResults)
                }
            }
        }
        //setup cancellation action on the continuation
        cont.invokeOnCancellation {
            context.unregisterReceiver(wifiScanReceiver)
        }
        //register broadcast reciever
        context.registerReceiver(wifiScanReceiver, IntentFilter(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION))
        //kick off scanning to eventually receive the broadcast
        wifiManager.startScan()
    }
这对于信号发射来说很好,但如果我想在扫描过程中获得结果,那么我会崩溃,因为 cont.resume()只能调用一次。然后我决定尝试Flow .这是我的代码:
suspend fun getCurrentScanResult(): Flow<List<ScanResult>> =
    flow{
        val wifiScanReceiver = object : BroadcastReceiver() {
            override fun onReceive(c: Context, intent: Intent) {
                if (intent.action?.equals(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION) == true) {
                    //context.unregisterReceiver(this)
                    emit(wifiManager.scanResults)
                }
            }
        }
        //setup cancellation action on the continuation
        //register broadcast reciever
        context.registerReceiver(wifiScanReceiver, IntentFilter(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION))
        //kick off scanning to eventually receive the broadcast
        wifiManager.startScan()
    }
但现在 Android Studio 说 Suspension functions can be called only within coroutine body功能 emit(wifiManager.scanResults)有没有办法在这里使用 Flow?

最佳答案

请查看callback flow这是专门为此用例设计的。像这样的东西可以完成这项工作:

callbackFlow {
  val receiver = object : BroadcastReceiver() {
    override fun onReceive(context: Context?, intent: Intent) {
      if (intent.action == WifiManager.SCAN_RESULTS_AVAILABLE_ACTION) {
        sendBlocking(wifiManager.scanResults) // or non-blocking offer()
      }
    }
  } 
  context.registerReceiver(receiver, intentFilter)

  awaitClose {
      context.unregisterReceiver(receiver)
  }
}
您可能还想与例如共享此流程。 shareIn运营商避免为每个流订阅者注册一个新的接收者。

关于android - 将广播接收器包装到流中(协程),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67071830/

相关文章:

android - 如何在android相机api上进行图像分析

java - IME 的自定义 Android View 的一部分被切断?

android - Recyclerview DiffUtil 项目更新

java - Android - 无法让广播接收器工作

java - 如何检测蓝牙设备是否超出范围,或者我们丢失了它?

android - ConstraintLayout 在 TextView 中右对齐文本

Android:无法打开数据库文件

android - 无法将 Intent 值传递给下一个 Activity

android - 存储库类中的协程范围

带有广播接收器的 Android Alarm Manager 在代码中而不是 list 中注册