在其静态构造函数中创建类的实例 - 为什么允许它?

dri*_*iis 6 c#

关于SO的另一个问题激发了我在C#中尝试这个代码:

class Program
{
    static Program() 
    {
        new Program().Run();
    }

    static void Main(string[] args) { }

    void Run()
    {
        System.Console.WriteLine("Running");
    }
}
Run Code Online (Sandbox Code Playgroud)

这会在运行时打印"正在运行".

我实际上期望编译器抱怨这个.毕竟,如果该类尚未被静态构造函数初始化; 我们怎样才能确定在它上面调用方法是否有效?

那么为什么编译器不会限制我们这样做呢?这有什么重要的使用场景吗?

编辑

我知道Singleton模式; 问题的关键是为什么我可以在静态构造函数完成之前调用实例上的方法.到目前为止,JaredPar的答案对此有一些很好的推理.

Jar*_*Par 6

问题略有不同.

编译器如何阻止你这样做?

当然,在样品中检测非常容易,但这个样品怎么样?

class Program {
  static void Fun() {
    new Program(); 
  }
  static Program() {
    Fun();
  }
}
Run Code Online (Sandbox Code Playgroud)

你可以欺骗编译器允许这样做的方式实际上是无止境的.即使编译器得到了所有的答案,你仍然可以用反射来击败它.

最后虽然这实际上是合法的,但如果有点危险,可以用C#和IL编写代码.只要你小心从这段代码中访问静态,这样做是安全的.对于像Singleton这样的某些模式,它也是有用的/可能是必要的


Han*_*ant 6

这是允许的,因为不允许这将是一个很大更糟.像这样的代码会严重陷入僵局:

class A {
    public static readonly A a;
    public static readonly B b;
    static A() {
        b = new B();
        a = B.a;
    }
}

class B {
    public static readonly A a;
    public static readonly B b;
    static B() {
        a = new A();
        b = A.b;
    }
}
Run Code Online (Sandbox Code Playgroud)

你当然是指着一把装满枪的脚.

CLI规范(Ecma 335)分区II,章节10.5.3.2"宽松保证"中记录了此行为:

可以使用属性beforefieldinit(第10.1.6节)标记类型,以指示§10.5.3.1中指定的保证不一定是必需的.特别是,不需要提供上面的最终要求:在调用或引用静态方法之前,不需要执行类型初始化程序.

[基本原理:当代码可以在多个应用程序域中执行时,确保最终保证变得特别昂贵.同时,对大量托管代码的检查表明,很少需要这种最终保证,因为类型初始化器几乎总是用于初始化静态字段的简单方法.将其留给CIL发生器(因此,可能还有程序员)来决定是否需要这种保证因此在需要时以一致性保证为代价提供效率.
最终理由]

C#编译器确实在类上发出了beforefieldinit属性:

.class private auto ansi beforefieldinit ConsoleApplication2.Program
       extends [mscorlib]System.Object
{
   // etc...
}
Run Code Online (Sandbox Code Playgroud)