- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
CIL 指令“Call”和“Callvirt”有什么区别?
最佳答案
当运行时执行call
指令时,它会调用一段确切的代码(方法)。毫无疑问它存在于何处。
IL 经过 JIT 处理后,调用站点生成的机器代码就是无条件
jmp
指令。
相比之下,callvirt
指令用于以多态方式调用虚拟方法。必须在每次调用运行时确定方法代码的确切位置。生成的 JITted 代码涉及通过 vtable 结构进行的一些间接操作。因此,调用执行速度较慢,但更灵活,因为它允许多态调用。
请注意,编译器可以发出虚拟方法的call
指令。例如:
sealed class SealedObject : object
{
public override bool Equals(object o)
{
// ...
}
}
考虑调用代码:
SealedObject a = // ...
object b = // ...
bool equal = a.Equals(b);
虽然System.Object.Equals(object)
是一个虚拟方法,但在这种用法中,无法存在Equals
方法的重载。 SealedObject
是一个密封类,不能有子类。
因此,.NET 的密封
类比非密封类具有更好的方法分派(dispatch)性能。
编辑:事实证明我错了。 C# 编译器无法无条件跳转到方法的位置,因为对象的引用(方法内 this
的值)可能为 null。相反,它会发出 callvirt
来执行 null 检查并在需要时抛出异常。
这实际上解释了我使用 Reflector 在 .NET 框架中发现的一些奇怪的代码:
if (this==null) // ...
编译器可能会发出 this
指针 (local0) 具有空值的可验证代码,只有 csc 不会这样做。
所以我猜 call
仅用于类静态方法和结构。
鉴于此信息,现在在我看来,sealed
仅对 API 安全有用。我发现another question这似乎表明密封您的类没有任何性能优势。
编辑 2:这比看起来要复杂得多。例如,以下代码发出 call
指令:
new SealedObject().Equals("Rubber ducky");
显然,在这种情况下,对象实例不可能为空。
有趣的是,在 DEBUG 构建中,以下代码会发出 callvirt
:
var o = new SealedObject();
o.Equals("Rubber ducky");
这是因为您可以在第二行设置断点并修改o
的值。在发布版本中,我认为调用将是 call
而不是 callvirt
。
不幸的是,我的电脑目前无法运行,但一旦它再次启动,我会尝试一下。
关于.net - 调用和 Callvirt,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/193939/
我试图了解 CLR 如何实现引用类型和多态性。我提到了 Don Box 的 Essential .Net Vol 1,它对校准大部分内容有很大帮助。但是当我尝试使用一些 IL 代码以更好地理解时,我被
CIL 指令“Call”和“Callvirt”有什么区别? 最佳答案 当运行时执行call指令时,它会调用一段确切的代码(方法)。毫无疑问它存在于何处。 IL 经过 JIT 处理后,调用站点生成的机器
是 il.EmitCall(OpCodes.Callvirt, getter, null); 等同于: 到 il.Emit(OpCodes.Callvirt,getter); 我不确定如何检查动态生成
如何确定一个方法是否需要使用“Call”或“Callvirt”来调用? 最佳答案 您可以一一遵循这些简单的规则来确定您应该使用哪一个: 方法是静态的吗?然后使用调用。 您正在调用的类型是值类型吗?然后
这个问题链接到How CLR calls methods correctly hidden by derived class when reference is stored in a base cl
在这种情况下,IL 并不总是对 virtual 方法使用 callvirt 指令: class MakeMeASandwich{ public override string ToString()
向某人解释虚拟调度很容易:每个对象都有一个指向表的指针作为其数据的一部分。类上有N个虚方法。每次调用特定方法时,我都会在对象到达时对其进行索引,并调用表中的第 i 个方法。每个实现方法 X() 的类都
在 IL 的一些实验中,我试图将程序集中的 callvirt 调用更改为 call 方法。基本上发生的事情是我有一个继承链,其中包含我正在调用的成员函数。 基本上调用与此类似: ((MyDerived
给定两种方法: static void M1(Person p) { if (p != null) { var p1 = p.N
对于下面的代码片段: struct Test { public override string ToString() { return ""; } } publ
我在其基类中标记为抽象的库类上调用属性集访问器。现在在运行时我 force应用程序针对另一个版本的库运行,其中类仅实现基类的底层接口(interface),但不是从它派生的。 有趣的是,.NET 将运
为什么编译器生成一个 callvirt 指令用于调用显式实现的接口(interface)方法和一个 call 用于调用 implicilty 在以下代码中实现了接口(interface)方法? 编译器
我有几个月前编写的这个框架,它生成一个类来调用此性能服务。框架的消费者使用方法创建一个接口(interface),使用属性进行注释,并调用一个工厂方法来创建他们可以用来调用此性能服务的接口(inter
我很想知道为什么会这样。请阅读下面的代码示例以及在每个部分下面的注释中发出的相应 IL: using System; class Program { static void Main()
我有一个类: public class SomeClass { public int I; public SomeClass(int input) { I = inpu
IL 提供了两种调用函数的语句,即 call 和 callvirt。 Call 用于调用非虚拟或静态函数或编译器不想对引用进行空检查的任何函数。 callvirt 用于调用虚函数,非虚函数也被调用,因
我有一个递归函数 emit : Map -> exp -> unit哪里il : ILGenerator对函数来说是全局的,并且 exp是一个判别联合,表示带有大小写 InstanceCall of
我是一名优秀的程序员,十分优秀!