我有一个Java Future
对象,我想转换成Scala Future
。查看j.u.c.Future
API文档,除了isDone
方法,似乎没有其他可用的选项。这个isDone
方法是否会阻塞程序?目前我在想:
val p = Promise()
if (javaFuture.isDone())
p.success(javaFuture.get)
有更好的方式吗?
我有一个Java Future
对象,我想转换成Scala Future
。查看j.u.c.Future
API文档,除了isDone
方法,似乎没有其他可用的选项。这个isDone
方法是否会阻塞程序?目前我在想:
val p = Promise()
if (javaFuture.isDone())
p.success(javaFuture.get)
有更好的方式吗?
从Scala 2.13
开始,标准库包括scala.jdk.FutureConverters
,该库提供了Java到Scala的CompletableFuture/Future
隐式转换:
import scala.jdk.FutureConverters._
// val javaFuture = java.util.concurrent.CompletableFuture.completedFuture(12)
val scalaFuture = javaFuture.asScala
// scalaFuture: scala.concurrent.Future[Int] = Future(Success(12))
如果只是简单地包装一下(我假设这里有一个隐式的ExecutionContext)呢:
val scalaFuture = Future {
javaFuture.get
}
编辑:
一个简单的轮询策略可以看起来像这样(java.util.Future => F):
def pollForResult[T](f: F[T]): Future[T] = Future {
Thread.sleep(500)
f
}.flatMap(f => if (f.isDone) Future { f.get } else pollForResult(f))
这将每500毫秒检查Java future是否完成。显然,总阻塞时间与上面相同(四舍五入到最近的500毫秒),但此解决方案将允许其他任务在ExecutionContext
的同一线程中交错执行。
Future {...}
中会神奇地消除任何阻塞。 - Régis Jean-GillesThread.sleep(500)
会像阻塞的javaFuture.get
调用一样阻塞同一线程,不是吗? - eddyP23sleep
会阻塞线程,但比 javaFuture.get
更可预测。一旦调用 get
,线程就会被阻塞直到完成。如果相同的 ExecutionContext
用于其他任务,则可能会阻塞几个长时间运行的 Futures。我并不是说轮询比仅包装 Future 更好,但在某些情况下可能是有意义的。 - pablochan对于现在阅读此问题的人,如果您正在使用Scala 2.13及以上版本,请使用:
import scala.jdk.FutureConverters._
使用completableFuture.asScala
进行转换
如果您正在使用Scala 2.12及以下版本,请使用
import scala.compat.java8.FutureConverters._
并且可以使用以下方式进行转换:toScala(completableFuture)
或 completableFuture.toScala
另外,在 Scala 2.12 中请确保您正在使用正确的构件:
org.scala-lang.modules:scala-java8-compat_2.12:0.9.0
如果由于某种原因,您手头的是 Future
,而不是现在很少见的 CompletableFuture
,请首先参考下面其中一个答案进行转换:将Java Future转换为CompletableFuture
import java.util.concurrent.CompletableFuture
import scala.compat.java8.FutureConverters
val javaFuture: java.util.concurrent.Future[Int] = ???
// Some method call which returns Java's Future[Int]
val scalaFuture: Future[Int] =
FutureConverters.toScala(CompletableFuture.supplyAsync(new Supplier[Int] {
override def get(): Int = javaFuture.get
}))
java.util.concurrent.CompletionStage
的子类型。它不适用于java.util.concurrent.Future
。 - Daniel Bimschas