我如何从Scalaz的Future中获得价值?

时间:2018-09-02 10:49:53

标签: scala future scalaz

我有以下代码:

package functorapplication

import scalaz._
import Scalaz._
import scalaz.concurrent.Future

object FunctorApplication2 extends App {

  val f1 = Future(3)//(ec)
  val f2 = Future(4)//(ec)
  val f3 = Future(5)//(ec)
  val calculate = (a: Int) => (b: Int) => (c: Int) => a + b + c
  val area = f1 <*> (f2 <*> (f3 <*> Future(calculate)))//(ec))) // Future(12)

  //println(area)//BindSuspend(scalaz.concurrent.Future$$Lambda...
  println("starting")
  val summed = for {
    a <- area
  } yield {
    println(a)
  }
  area.map(value => println(value))

  //println(summed)//Suspend(scalaz.concurrent.Future$$Lambda...
  println("done")

}

这将产生以下结果:

starting
done

关键是-未来似乎没有价值可以理解或映射。

我的问题是:我如何从Scalaz的Future中获得价值?

注意:

这是我的scala版本

scalaVersion := "2.12.5",

这是我的scalaz版本

  "org.scalaz" %% "scalaz-core" % "7.2.26",
  "org.scalaz" %% "scalaz-concurrent" % "7.2.26",
  "org.scalaz" %% "scalaz-effect" % "7.2.26",
  "org.scalaz" %% "scalaz-iteratee" % "7.2.26"

2 个答案:

答案 0 :(得分:0)

虽然我对scalaz Future知之甚少,但我认为您只是缺少.start.unsafeStart来真正开始运行Future

即使您使用.start,也需要使用scala.io.StdIn.readLine("press enter to exit")之类的东西来阻塞主线程。

另一种选择是使用.rununsafePerformSync,它们将阻止并等待结果。

val res = summed.run
println(res) // currently Unit because it yields println
println("done")

这将产生副作用,并运行Future并返回结果。

我通过查看scalaz.concurrent.Future的源代码找到了所有这一切。

您应该注意,实际上,由于错误处理,人们会使用Task,除非您正在编写库(如源代码中所述)。

答案 1 :(得分:0)

如果您拥有Future[A]类型的值,那么您有资格在将来的某个时间获取

  • 值为A的值
  • 或一个错误,表示为类型Throwable的值。

因此,未来代表着潜在的未来价值。唯一获得未来价值的方法是等待

import scala.concurrent._
import scala.concurrent.ExecutionContext.Implicits.global
import scala.concurrent.duration._

val f1 = Future(3)//(ec)
val f2 = Future(4)//(ec)
val f3 = Future(5)//(ec)

val calculate = (a: Int) => (b: Int) => (c: Int) => a + b + c

val area: Future[Int] = for {
  v1 <- f1 
  v2 <- f2 
  v3 <-f3 
} yield calculate(v1)(v2)(v3)

println("starting")

println("adding future side effect")
//this happens in the future
val withSideEffect = area.map(value => println("side-effect: " + value))

println("awaiting now")
//now, let's wait
println("Await: " + Await.ready(withSideEffect, Duration.Inf).value)

println("done")

此打印:

starting
adding future side effect
awaiting now
side-effect: 12
Await: Some(Success(()))
done