module Lazy:sig
..end
延迟计算。
type'a
t ='a CamlinternalLazy.t
类型为 'a Lazy.t
的值是一个延迟计算,称为挂起,其结果类型为 'a
。特殊的表达式语法 lazy (expr)
创建一个计算 expr
的挂起,但不会立即计算 expr
本身。“强制”挂起将计算 expr
并返回其结果。使用特殊模式语法 lazy(pattern)
匹配挂起也会计算底层表达式并尝试将其绑定到 pattern
let lazy_option_map f x =
match x with
| lazy (Some x) -> Some (Lazy.force f x)
| _ -> None
注意:如果延迟模式出现在模式匹配中的多个情况中,则即使在最终由模式匹配选择的 case 之外,也可能会强制延迟表达式。在上面的例子中,挂起 x
始终被计算。
注意:lazy_t
是编译器为 lazy
关键字使用的内置类型构造函数。您不应该直接使用它。始终使用 Lazy.t
代替。
注意:Lazy.force
不是并发安全的。如果您将此模块与多个 fiber、systhread 或 domain 结合使用,则需要添加一些锁。但是,该模块确保内存安全,因此,并发访问此模块不会导致崩溃,但行为未定义。
注意:如果程序使用 -rectypes
选项编译,则形式为 let rec x = lazy x
或 let rec x = lazy(lazy(...(lazy x)))
的无根据递归定义将被类型检查器接受,并且在强制时会导致错误的值,这些值会在垃圾收集器和运行时系统其他部分中触发无限循环。没有 -rectypes
选项,这种无根据递归定义将被类型检查器拒绝。
exception Undefined
当从多个 fiber、systhread 或 domain 并发强制挂起,或者挂起尝试递归强制自身时引发。
val force : 'a t -> 'a
force x
强制挂起 x
并返回其结果。如果 x
已经强制,则 Lazy.force x
将再次返回相同的值,而不会重新计算它。如果它引发了异常,则会再次引发相同的异常。
Undefined
(见 Lazy.Undefined
)。val map : ('a -> 'b) -> 'a t -> 'b t
map f x
返回一个挂起,当强制时,它会强制 x
并将其值应用于 f
。
它等效于 lazy (f (Lazy.force x))
。
val is_val : 'a t -> bool
is_val x
如果 x
已经强制并且没有引发异常,则返回 true
。
val from_val : 'a -> 'a t
from_val v
首先评估 v
(就像任何函数一样)并返回其结果的已强制挂起。它与 let x = v in lazy x
相同,但在某些情况下使用动态测试来优化挂起创建。
val map_val : ('a -> 'b) -> 'a t -> 'b t
map_val f x
如果 x
已经强制,则直接应用 f
,否则它与 map f x
行为相同。
当 x
已经强制时,此行为可以节省挂起的构造,但另一方面,它会更积极地执行更多工作,如果从未强制函数结果,则这些工作可能没有用。
如果 f
引发异常,则将在 is_val x
时立即引发,或者仅在强制 thunk 时引发。
如果 map_val f x
没有引发异常,则 is_val (map_val f x)
等于 is_val x
。
以下定义仅供高级使用;它们需要熟悉延迟编译方案才能正确使用。
val from_fun : (unit -> 'a) -> 'a t
from_fun f
与 lazy (f ())
相同,但效率略高。
它应该仅在函数 f
已定义时使用。特别地,编写 from_fun (fun () -> expr)
始终比 lazy expr
效率低。
val force_val : 'a t -> 'a
force_val x
强制挂起 x
并返回其结果。如果 x
已经强制,则 force_val x
将再次返回相同的值,而不会重新计算它。
如果计算 x
引发了异常,则 force_val x
是否引发相同的异常或 Lazy.Undefined
是未定义的。
Undefined
如果 x
的强制尝试递归强制 x
本身。Undefined
(见 Lazy.Undefined
)。