JIT 在 .NET 中的工作方式是,在方法被 jit 之前,方法表入口指向一个小存根,该存根将在调用时 JIT 方法。之后更新方法表以引用 JIT 编译代码的位置。
鉴于只有被调用的方法是 JIT 编译的,所以没有被调用的方法没有 JIT 开销。
JIT 编译器将在需要时编译整个方法。如果是发布版本,则可能会优化构建代码,否则将完整编译该方法。
您可以使用 WinDbg/SOS 检查方法表。考虑以下:
class SomeType
{
public void Called()
{
Console.WriteLine("called");
}
public void NotCalled()
{
Console.WriteLine("not called");
}
}
假设我们创建一个实例SomeType
,调用Called
然后检查方法表SomeType
。在 x86 上,您会看到如下内容:
0:000> !dumpmt -md 00a7381c
EEClass: 00a712d0
Module: 00a72e94
Name: ConsoleApplication1.SomeType
mdToken: 02000002
File: c:\temp\ConsoleApplication1\ConsoleApplication1\bin\Debug\ConsoleApplication1.exe
BaseSize: 0xc
ComponentSize: 0x0
Slots in VTable: 7
Number of IFaces in IFaceMap: 0
--------------------------------------
MethodDesc Table
Entry MethodDe JIT Name
72ca4960 729a6728 PreJIT System.Object.ToString()
72c98790 729a6730 PreJIT System.Object.Equals(System.Object)
72c98360 729a6750 PreJIT System.Object.GetHashCode()
72c916f0 729a6764 PreJIT System.Object.Finalize()
00df00d8 00a73814 JIT ConsoleApplication1.SomeType..ctor()
00df0110 00a737fc JIT ConsoleApplication1.SomeType.Called()
00a7c031 00a73808 NONE ConsoleApplication1.SomeType.NotCalled()
请注意,它Called
是 JIT 编译的,但由于我们还没有调用NotCalled
它,所以它还没有被 JIT 编译。
另外,请注意,来自的方法object
都已经过 PreJIT 编译。
请记住,在发布构建中,短方法可能会被内联,在这种情况下,它们不会被称为方法,而只是作为调用站点生成代码的一部分包含在内。