前言
我们知道,在调用构造函数时,C#会先对类中的字段、属性进行初始化操作。那么,问题来了,为什么在调用构造函数前会初始化类中的字段和属性呢?让我们一起通过ildasm来揭开构造函数的面纱吧。
需要反编译的C#代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
|
class CtorTester { private string _name; private int _age = 10; public int Age { get ; set ; } = 20; public CtorTester() { _name = "Name" ; } } |
使用ildasm工具对.exe文件进行IL反编译,下面是构造函数的IL代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
.method public hidebysig specialname rtspecialname instance void .ctor() cil managed { // Code size 36 (0x24) .maxstack 8 IL_0000: ldarg.0 //将索引为 0 的参数压栈。 IL_0001: ldc.i4.s 10 //将int 10压栈。 IL_0003: stfld int32 _011_Ctor.CtorTester::_age //将栈顶的值赋给栈中第二个值,即_age=10,完成字段_age的初始化操作。 IL_0008: ldarg.0 IL_0009: ldc.i4.s 20 IL_000b: stfld int32 _011_Ctor.CtorTester:: '<Age>k__BackingField' //完成属性Age的初始化操作。 IL_0010: ldarg.0 IL_0011: call instance void [mscorlib]System.Object::.ctor() //调用基类Object的构造函数 IL_0016: nop IL_0017: nop IL_0018: ldarg.0 IL_0019: ldstr "Name" IL_001e: stfld string _011_Ctor.CtorTester::_name //完成字段_name的赋值操作,即构造函数中的“_name = "Name";” IL_0023: ret } // 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
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:http://www.cnblogs.com/tianlang358/p/10369763.html