通过IL了解C#类的构造函数

我们知道,在调用构造函数时,C#会先对类中的字段、属性进行初始化操作。那么,问题来了,为什么在调用构造函数前会初始化类中的字段和属性呢?让我们一起通过ildasm来揭开构造函数的面纱吧。

需要反编译的C#代码如下:

 1 class CtorTester
 2 {
 3     private string _name;
 4     private int _age = 10;
 5 
 6     public int Age { get; set; } = 20;
 7 
 8     public CtorTester()
 9     {
10         _name = "Name";
11     }
12 }

使用ildasm工具对.exe文件进行IL反编译,下面是构造函数的IL代码:

 1 .method public hidebysig specialname rtspecialname 
 2         instance void  .ctor() cil managed
 3 {
 4   // Code size       36 (0x24)
 5   .maxstack  8
 6   IL_0000:  ldarg.0 //将索引为 0 的参数压栈。
 7   IL_0001:  ldc.i4.s   10 //将int 10压栈。
 8   IL_0003:  stfld      int32 _011_Ctor.CtorTester::_age //将栈顶的值赋给栈中第二个值,即_age=10,完成字段_age的初始化操作。
 9   IL_0008:  ldarg.0
10   IL_0009:  ldc.i4.s   20
11   IL_000b:  stfld      int32 _011_Ctor.CtorTester::'<Age>k__BackingField' //完成属性Age的初始化操作。
12   IL_0010:  ldarg.0
13   IL_0011:  call       instance void [mscorlib]System.Object::.ctor() //调用基类Object的构造函数
14   IL_0016:  nop
15   IL_0017:  nop
16   IL_0018:  ldarg.0
17   IL_0019:  ldstr      "Name"
18   IL_001e:  stfld      string _011_Ctor.CtorTester::_name //完成字段_name的赋值操作,即构造函数中的“_name = "Name";”
19   IL_0023:  ret
20 } // end of method CtorTester::.ctor

通过对构造函数的IL代码的解析发现,C#在编译时会将对字段、属性的初始化内联到构造函数中。这也就是为什么我们在调用构造函数之前会先执行字段、属性初始化代码的原因。

 

ildasm.exe工具的存放路径:
C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin

参考资料
IL指令说明:https://docs.microsoft.com/zh-cn/dotnet/api/system.reflection.emit.opcodes?
redirectedfrom=MSDN&view=netframework-4.7.2#fields

posted @ 2019-02-13 14:44  叶落劲秋  阅读(426)  评论(0编辑  收藏  举报