将Java Future转换为Scala Future

时间:2016-01-27 09:23:26

标签: scala future scala-java-interop

我有一个Java Future对象,我想将其转换为Scala Future

查看j.u.c.Future API,除了isDone方法之外,没有什么可以使用的。此isDone方法是否会阻止?

目前这就是我的想法:

val p = Promise()
if (javaFuture.isDone())
  p.success(javaFuture.get)

有更好的方法吗?

4 个答案:

答案 0 :(得分:10)

如何包装它(我假设这里有一个隐含的ExecutionContext):

def pollForResult[T](f: F[T]): Future[T] = Future {
    Thread.sleep(500)
    f
  }.flatMap(f => if (f.isDone) Future { f.get } else pollForResult(f))

编辑:

简单的轮询策略可能如下所示(java.util.Future => F):

ExecutionContext

这将检查Java未来是否每500毫秒完成一次。显然,总阻塞时间与上面相同(向上舍入到最接近的500ms),但是这个解决方案将允许其他任务在{{1}}的同一个线程中交错。

答案 1 :(得分:6)

Scala 2.13开始,标准库包括scala.jdk.FutureConverters,该库提供了Java至Scala CompletableFuture/Future的隐式转换:

import scala.jdk.FutureConverters.Ops._

// val javaFuture = java.util.concurrent.CompletableFuture.completedFuture(12)
val scalaFuture = javaFuture.asScala
// scalaFuture: scala.concurrent.Future[Int] = Future(Success(12))

答案 2 :(得分:5)

使用FutureConvertors(Scala中的内置实用程序)将Java Future转换为Scala Future。

考虑将Java Future [Int]转换为Scala Future [Int]的示例::

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 
    }))

类似地,我们可以为任何自定义类型(而不是Int)做

答案 3 :(得分:1)

对于那些现在阅读此问题的人,如果您使用的是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(在当今罕见的情况下),请遵循以下答案中的第一个:Transform Java Future into a CompletableFuture < / p>