我正在开发一个基于Java的Android应用程序。我想开始在主线程之外使用Kotlin & coroutines进行与网络相关的工作,但是我遇到了一个问题,试图弄清楚如何成功地处理某个特定的情况。
我已经创建了一个带有网络函数的Kotlin文件,我计划在Java类(视图模型,此时需要用Java编写)中调用它。这个想法是,我希望网络函数修改并将列表返回给视图模型,然后允许视图模型继续。
fun performNetworking(
data: MutableList<Data>
): MutableList<Data> {
CoroutineScope(Dispatchers.IO).launch {
data.forEach {
// modify each element in the list
}
}
return data
}目前,我从Java视图模型调用它,如下所示:
public void modifyData(List<Data> data) {
List<Data> modifiedData = NetworkingKt.performNetworking(data);
performMoreThings(modifiedData);
}使用此实现,performMoreThings()将触发,就好像列表中的任何数据都未被修改一样。通过调试,我发现在协程作用域可以做任何事情之前,performMoreThings()的结论是通过未修改的数据列表得出的。
因此,我想知道是否有一种方法可以让代码在将数据列表返回到Java视图模型之前等待协程作用域的工作完成(而不锁定主线程并阻止用户与应用程序交互)?或者我误解了协程的功能、目的和实现,而这是不可能的?我是否必须传入对视图模型的引用,并在协程完成后让它调用延续方法,就像我在onPostExecute()中处理异步任务一样
发布于 2021-11-02 20:32:18
不,这不可能。不是因为协程的局限性,而是相反的--由于非协程代码的局限性。您的modifyData()是一个常规的、不可挂起的方法,所以它无论如何都不能在不阻塞线程的情况下等待。Kotlin的暂停函数可以做到这一点,但是Java方法不能做到这一点。
您需要从后台线程调用modifyData(),这样阻塞就不会成为问题,或者重新设计performNetworking()以使用处理异步任务的经典技术之一,即接收回调或返回未来:
fun performNetworking(
data: MutableList<Data>
): CompletableFuture<MutableList<Data>> {
return CoroutineScope(Dispatchers.IO).future {
data.forEach {
// modify each element in the list
}
data
}
}public void modifyData(List<Data> data) {
NetworkingKt.performNetworking(data)
.thenAccept(this::performMoreThings);
}https://stackoverflow.com/questions/69816394
复制相似问题