模块 Lazy

module Lazy: sig .. end

延迟计算。


type 'a t = 'a CamlinternalLazy.t 

类型为 '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 xlet rec x = lazy(lazy(...(lazy x))) 的无根据递归定义将被类型检查器接受,并且在强制时会导致错误的值,这些值会在垃圾收集器和运行时系统其他部分中触发无限循环。没有 -rectypes 选项,这种无根据递归定义将被类型检查器拒绝。

exception Undefined

当从多个 fiber、systhread 或 domain 并发强制挂起,或者挂起尝试递归强制自身时引发。

val force : 'a t -> 'a

force x 强制挂起 x 并返回其结果。如果 x 已经强制,则 Lazy.force x 将再次返回相同的值,而不会重新计算它。如果它引发了异常,则会再次引发相同的异常。

迭代器

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 flazy (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 是未定义的。