10

我有以下 F# 程序:

open MyModule

printfn "%d" test

MyModule 是:

module MyModule

printfn "foo"

let test = 
  printfn "bar"
  42

这会产生以下输出:

foo
bar
42

当我将 MyModule 更改为:

module MyModule

printfn "foo"

let test = 
  // printfn "bar" <-- note the comment!
  42

...结果是:

42

为什么不再打印“foo”?

4

1 回答 1

18

我认为规范的第 12.5.1 节Execution of Static Initializers有你的答案。引用相关位:

文件的静态初始化程序在首次访问具有可观察初始化的值时执行

除了模块中的以下定义外,所有定义都具有可观察的初始化:

以下列表包括:

绑定到简单常量表达式的非可变、非线程局部值

注释掉 的第一行后test,就变成了一个常量表达式。因此,它不再触发静态初始化。

编辑

规范没有提供这种行为的基本原理,但它类似于 C#。例如,在这段代码中,静态初始化永远不会发生:

class Program {
    static void Main(string[] args) {
        Console.WriteLine(T.Integer);
        Console.WriteLine(T.Null);
        Console.WriteLine(T.Enum);
        Console.Read();
    }
}

static class T {
    static T() {
        Console.WriteLine("You won't see this.");
    }
    public const int Integer = 1;
    public const string Null = null;
    public const ConsoleKey Enum = ConsoleKey.Escape;
}
于 2013-07-10T16:08:08.417 回答