基本上我正在cassandra上運行兩個期貨查詢,然後我需要做一些計算並返回值(平均值)。卡住與斯卡拉期貨
這裏是我的代碼:
object TestWrapFuture {
def main(args: Array[String]) {
val category = 5392
ExtensiveComputation.average(category).onComplete {
case Success(s) => println(s)
case Failure(f) => throw new Exception(f)
}
}
}
class ExtensiveComputation {
val volume = new ListBuffer[Int]()
def average(categoryId: Int): Future[Double] = {
val productsByCategory = Product.findProductsByCategory(categoryId)
productsByCategory.map { prods =>
for (prod <- prods if prod._2) {
Sku.findSkusByProductId(prod._1).map { skus =>
skus.foreach(sku => volume += (sku.height.get * sku.width.get * sku.length.get))
}
}
val average = volume.sum/volume.length
average
}
}
}
object ExtensiveComputation extends ExtensiveComputation
那麼是什麼問題?
skus.foreach將結果值附加到ListBuffer中。因爲一切都是異步的,所以當我嘗試在我的主體中獲得結果時,我得到一個錯誤,說我不能被零除。
事實上,由於我的Sku.findSkusByProduct返回一個Future,當我嘗試計算平均值時,該卷是空的。
我應該在計算之前阻止任何事情,還是應該做其他事情?
編輯
嗯,我試圖阻止這樣的:
val volume = new ListBuffer[Int]()
def average(categoryId: Int): Future[Double] = {
val productsByCategory = Product.findProductsByCategory(categoryId)
val blocked = productsByCategory.map { prods =>
for (prod <- prods if prod._2) {
Sku.findSkusByProductId(prod._1).map { skus =>
skus.foreach(sku => volume += (sku.height.get * sku.width.get * sku.length.get))
}
}
}
Await.result(blocked, Duration.Inf)
val average = volume.sum/volume.length
Future.successful(average)
}
然後我從這段代碼兩種不同的結果:
Sku.findSkusByProductId(prod._1).map { skus =>
skus.foreach(sku => volume += (sku.height.get * sku.width.get * sku.length.get))
}
1 - 當有隻有幾個像cassandra擡頭看50,它只是運行,並給我結果
2 - 當有許多像1000,它給了我
java.lang.ArithmeticException:/零
EDIT 2
我想這個代碼@Olivier Michallat提議
def average(categoryId: Int): Future[Double] = {
val productsByCategory = Product.findProductsByCategory(categoryId)
productsByCategory.map { prods =>
for (prod <- prods if prod._2) findBlocking(prod._1)
volume.sum/volume.length
}
}
def findBlocking(productId: Long) = {
val future = Sku.findSkusByProductId(productId).map { skus =>
skus.foreach(sku => volume += (sku.height.get * sku.width.get * sku.length.get))
}
Await.result(future, Duration.Inf)
}
而下面這個作爲@kolmar的提議:
def average(categoryId: Int): Future[Int] = {
for {
prods <- Product.findProductsByCategory(categoryId)
filtered = prods.filter(_._2)
skus <- Future.traverse(filtered)(p => Sku.findSkusByProductId(p._1))
} yield {
val volumes = skus.flatten.map(sku => sku.height.get * sku.width.get * sku.length.get)
volumes.sum/volumes.size
}
}
兩項工程的幾個單品找到像50,但都失敗了許多的SKU找到像1000投擲ArithmeticException:/零
看來,它不能返回未來之前計算的一切...
請結帳我的更新... –