我目前正在尝试更多地利用kotlin协程。但我面临一个问题:当在这些协程中使用moshi或okhttp时,我得到一个警告:

“不恰当的阻塞方法调用”

解决这些问题的最好方法是什么?我真的不想不合适;-)


当前回答

这就是如何挂起协程,在线程中运行阻塞方法,并在结果上恢复它。这也将处理异常,所以你的应用程序不会崩溃。

suspendCoroutine { continuation ->
    thread {
        try {
            doHttpRequest(URL(...)) {
                continuation.resume(it)
            }
        }
        catch (t: Throwable) {
            continuation.resumeWithException(t)
        }
    }
}

编辑:预期的方法是使用witchContext(Dispatchers.IO)。我把这个回答留在这里,以防有人发现这种方法有用。

其他回答

使用withContext将“不适当的阻塞方法调用”代码包装到另一个上下文中。

也就是说(例如):

如果你正在执行一个读/写阻塞方法调用:

val objects = withContext(Dispatchers.IO) { dao.getAll() }

如果您正在执行阻塞网络请求(使用Retrofit):

val response = withContext(Dispatchers.IO) { call.execute() }

或者如果你正在执行一个CPU密集型的阻塞任务:

val sortedUsers = withContext(Dispatchers.Default) { users.sortByName() }

这将挂起当前的协程,然后在不同的线程上执行“不适当的阻塞调用”(来自Dispatchers)。IO或Dispatchers。(默认池),因此不会阻塞协程正在执行的线程。

如果你选择像一些答案建议的那样压抑,那就使用

@Suppress(“BlockingMethodInNonBlockingContext”)

它看起来像在kotlin.runCatching()中封装调用解决了警告,但不确定为什么…因为正如之前关于runCatching的回答所说,这不是由于抛出异常,因为即使try{} catch也不能解决问题,可能是一些错误检测问题… 我现在使用了下面的方法……

val result = kotlin.runCatching {
     OldJavaLib.blockingCallThatThrowsAnException()
}
if (result.isSuccess) {
      print("success is on your side")
} else {
      print("one failure is never the end")
}

此警告是关于阻塞当前线程和协程的方法不能正确挂起。这样,您就失去了协程的所有好处,并再次降级为每个线程一个任务。

每一种情况都应以不同的方式处理。对于可悬挂的http调用,您可以使用ktor http客户端。但有时没有针对您的情况的库,因此您可以编写自己的解决方案,或者忽略此警告。

编辑:withContext(Dispatchers.IO)或一些自定义调度程序可以用来解决这个问题。谢谢你的评论。

我使用Android Studio 4.1,当我使用Moshi或操作文件时,警告显示。即使我确定我在做什么,在withContext中包装代码也没有帮助。

我最近发现,将警告的小代码移动到标准方法中,而不像有趣的action(){…}可以删除警告。这很难看,因为它只是隐藏了警告。

更新:从我的个人经验来看,似乎抑制警告或runBlocking更直接。