我在
LINQPad中一直在玩一些C#语句,以便了解发出的中间语言代码.
我首先尝试了以下代码:
var Container = new {Name = "James"}; Console.WriteLine(Container.Name);
并且看到以下六行IL发射:
IL_0001: ldstr "James" IL_0006: newobj <>f__AnonymousType0<System.String>..ctor IL_000B: stloc.0 IL_000C: ldloc.0 IL_000D: callvirt <>f__AnonymousType0<System.String>.get_Name IL_0012: call System.Console.WriteLine
这是我所期望的,并且非常好地演示了匿名类型是如何只读/不可变的,因为没有set_Name属性.
接下来我尝试了这些陈述:
dynamic Container = new System.Dynamic.ExpandoObject(); Container.Name = "James"; Console.WriteLine(Container.Name);
这会导致大量的IL被释放.我不会在这里粘贴它,但你可以在this pastebin找到它.
我理解在管理动态类型和ExpandoObject方面存在相当多的开销,但我不明白为什么在这种情况下通过内部反射执行对System.Console.WriteLine的调用.
IL_0072: ldstr "WriteLine" .... IL_00BF: ldtoken System.Console
在第一段代码中,在检索并存储属性之后,它是一个调用System.Console.WriteLine的单行IL语句.
那么为什么动态类型的呼叫需要额外的所有这些?