- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
Scala 的 Future
( new in 2.10 和 now 2.9.3 )是一个应用仿函数,这意味着如果我们有一个 traversable type F
,我们可以将一个 F[A]
和一个函数 A => Future[B]
并将它们转换为一个 Future [F[B]]
。
此操作在标准库中可用 Future.traverse
。 Scalaz 7还提供了一个更通用的遍历
,如果我们从scalaz-contrib
library导入Future
的应用仿函数实例,我们可以在这里使用它。 .
这两个traverse
方法在流的情况下表现不同。标准库遍历在返回之前消耗流,而 Scalaz 的 returns the future immediately :
import scala.concurrent._
import ExecutionContext.Implicits.global
// Hangs.
val standardRes = Future.traverse(Stream.from(1))(future(_))
// Returns immediately.
val scalazRes = Stream.from(1).traverse(future(_))
还有另一个区别,如 Leif Warner观察here 。标准库的 traverse
立即启动所有异步操作,而 Scalaz 则启动第一个操作,等待其完成,启动第二个操作,等待它,依此类推。
通过编写一个函数,为流中的第一个值休眠几秒钟,可以很容易地显示第二个差异:
def howLong(i: Int) = if (i == 1) 10000 else 0
import scalaz._, Scalaz._
import scalaz.contrib.std._
def toFuture(i: Int)(implicit ec: ExecutionContext) = future {
printf("Starting %d!\n", i)
Thread.sleep(howLong(i))
printf("Done %d!\n", i)
i
}
现在 Future.traverse(Stream(1, 2))(toFuture)
将打印以下内容:
Starting 1!
Starting 2!
Done 2!
Done 1!
以及 Scalaz 版本 (Stream(1, 2).traverse(toFuture)
):
Starting 1!
Done 1!
Starting 2!
Done 2!
这可能不是我们想要的。
奇怪的是,这两个遍历在列表上的行为在这方面是相同的 - Scalaz 不会等到一个 future 完成后再开始下一个。
Scalaz 还包括自己的 concurrent
包及其自己的 future 实现。我们可以使用与上面相同的设置:
import scalaz.concurrent.{ Future => FutureZ, _ }
def toFutureZ(i: Int) = FutureZ {
printf("Starting %d!\n", i)
Thread.sleep(howLong(i))
printf("Done %d!\n", i)
i
}
然后我们得到了 Scalaz 在流列表以及流上的行为:
Starting 1!
Done 1!
Starting 2!
Done 2!
也许不那么令人惊讶的是,遍历无限流仍然会立即返回。
此时我们确实需要一个表格来总结,但列表也可以:
还有:
这有什么意义吗?在列表和流上的此操作是否有“正确”的行为?是否有某种原因导致“最异步”行为(即,在返回之前不消耗集合,并且在继续下一个之前不等待每个 future 完成)未在此处表示?
最佳答案
我无法全部回答,但我尝试了一些部分:
Is there some reason that the "most asynchronous" behavior—i.e., don't consume the collection before returning, and don't wait for each future to complete before moving on to the next—isn't represented here?
如果您有相关计算且线程数量有限,则可能会遇到死锁。例如,您有两个 future 依赖于第三个 future(future 列表中的所有三个),并且只有两个线程,您可能会遇到前两个 future 阻塞所有两个线程而第三个线程永远不会执行的情况。 (当然,如果你的池大小是1,即执行一次计算后执行另一次计算,你也会得到类似的情况)
为了解决这个问题,每个 future 都需要一个线程,没有任何限制。这适用于小型 future list ,但不适用于大型 future list 。因此,如果您并行运行所有程序,您将遇到这样一种情况:小示例在所有情况下都会运行,而较大的示例将陷入僵局。 (示例:开发人员测试运行良好,生产死锁)。
Is there a "correct" behavior for this operation on lists and streams?
我认为对于 future 来说这是不可能的。如果您了解更多的依赖关系,或者当您确定计算不会阻塞时,可能会出现更并发的解决方案。但执行 future list 对我来说是“被设计破坏的”。最好的解决方案似乎是一个,对于死锁的小例子来说,它已经失败了(即执行一个 Future 之后另一个)。
Scalaz futures with lists: do wait for each future to complete.
我认为 scalaz 在内部使用理解来进行遍历。对于推导式,不能保证计算是独立的。所以我猜想 Scalaz 在理解方面做了正确的事情:一个接一个地进行计算。对于 future,这将始终有效,因为您的操作系统中有无限的线程。
换句话说:您看到的只是理解式(必须)如何工作的产物。
我希望这是有道理的。
关于scala - 使用返回 future 的函数遍历列表和流,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18163656/
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!