public sealed class Program{ public static void Main() { System.Console.WriteLine("Hi"); } }
程序运行时,CLR会加载并初始化它。CLR读取程序集的CLR头,查找标识了应用程序入口方法(Main)的MethodDefToken。然后,CLR会检索MethodDef元数据表,找到该方法的IL代码在文件中的偏移量,把这些IL代码JIT(just-in-time)编译成本地(native)代码。编译时会对代码进行验证以确保类型安全性。最后,将执行本地代码。下面展示了Main方法的IL代码。(运行ILDasm.exe可以生成)
.method public hidebysig static void Main() cil managed //SIG: 00 00 01 { .entrypoint //Method begins at RVA 0x2050 //Code size 11 (0xb) .maxstack 8 IL_0000: /* 72 | (70)000001 */ ldstr "Hi" IL_0005: /* 28 | (0A)000003 */ call void [mscorlib]System.Console::WriteLine(string) IL_00a: /* 2A | */ ret }//end of method Program::Main
对这个代码进行JIT编译时,CLR会检查对类型和成员的所有引用,并加载定义了它们的程序集(如果尚未加载)。可以看出,上述IL代码有一个对System.Console.WriteLine的引用。具体地说,IL call 指令应用了token 0A000003.这个token对应于MemberRef源数据表中的记录项3.CLR检查这个MemberRef记录项,发现它的一个字段引用了一个TypeRef表中的一个记录项(System.Console类型)。根据TypeRef记录项,CLR被引导至一个AssemblyRef记录项:“mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089”.这样,CLR就知道了它需要的是哪一个程序集。接着,CLR必须定位并加载该程序集。
解析一个引用的类型时,CLR可能在以下三个地方找到类型。
- 同一个文件
- 不同的文件,同一个程序集
- 不同的文件,不同的程序集
下图为一个类型绑定的图解过程: