代码之家  ›  专栏  ›  技术社区  ›  Juliet

计算表达式不执行let

  •  3
  • Juliet  · 技术社区  · 16 年前

    我使用的是f v 1.9.6.2,我定义了一个非常简单的计算表达式:

    type MaybeBuilder() =
        member this.Let(x, f) =
            printfn "this.Let: %A" x
            this.Bind(Some x, f)
        member this.Bind(x, f) =
            printfn "this.Bind: %A" x
            match x with
            | Some(x) when x >= 0 && x <= 100 -> f(x)
            | _ -> None
        member this.Delay(f) = f()
        member this.Return(x) = Some x
    
    let maybe = MaybeBuilder()
    

    我在代码中撒了一些print语句,告诉我在计算表达式中调用了哪些方法。当我执行以下语句时:

    maybe {
        let x = 12
        let! y = Some 11
        let! z = Some 30
        return x + y + z
    }
    

    我希望控制台打印出以下内容:

    this.Let 12
    this.Bind Some 12
    this.Bind Some 11
    this.Bind Some 30

    但我的实际结果如下:

    this.Bind: Some 11
    this.Bind: Some 30

    换句话说,f似乎没有执行 Let 成员。当我重写 要抛出异常,代码运行时没有异常。另外,当我评论 全体成员,我 获取一条错误消息,说明 The field, constructor or member 'Let' is not defined ,代码按预期执行。

    (我试过使用Reflector来研究代码,但通常情况下,反编译的f会被破坏,超出可读性。)

    看起来像 spec for computation expressions 改变了。是 let 绑定不再被视为语法糖,而是 计算工作流中不再需要成员?

    3 回复  |  直到 8 年前
        1
  •  4
  •   Abel    8 年前

    你自己也有答案。从 F# spec 它描述了如何转换计算表达式:

    {| let binds in cexpr |}C  = let binds in {| cexpr |}C)
    

    所以不,您不再需要显式地定义let了,它是由编译器翻译的。

    更新:在 the detailed release notes of the September CTP .

        2
  •  0
  •   MichaelGG    16 年前

    正确--您不能再为let()提供绑定。

        3
  •  0
  •   Brian    16 年前
    推荐文章