
class Program{ static Program() { new Program().Run(); } static voID Main(string[] args) { } voID Run() { System.Console.Writeline("Running"); }} 这会在运行时打印“正在运行”.
我实际上期望编译器抱怨这个.毕竟,如果该类尚未被静态构造函数初始化;我们怎样才能确定在它上面调用方法是否有效?
那么为什么编译器不会限制我们这样做呢?这有什么重要的使用场景吗?
编辑
我知道Singleton模式;问题的关键是为什么我可以在静态构造函数完成之前调用实例上的方法.到目前为止,JaredPar的答案对此有一些很好的推理.
解决方法 这是允许的,因为不允许它会更糟糕.像这样的代码会严重陷入僵局: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; }} 你当然是指着一把装满q的脚.
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...} 总结 以上是内存溢出为你收集整理的c# – 在其静态构造函数中创建类的实例 – 为什么允许它?全部内容,希望文章能够帮你解决c# – 在其静态构造函数中创建类的实例 – 为什么允许它?所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)