我试图理解继承在C#中是如何工作的。我写了以下代码:
class Program
{
static void Main(string[] args)
{
Animal animal = new Dog();
animal.OverRideMe();
//animal.NewMethod();
Dog dog = (Dog)animal;
dog.OverRideMe();
dog.NewMethod();
Console.Read();
}
}
public abstract class Animal
{
public Animal()
{
Console.WriteLine("Base Constructor");
}
public virtual void OverRideMe()
{
Console.WriteLine("In Base Class's OverRideMe");
Console.Read();
}
}
public class Dog : Animal
{
public Dog()
{
Console.WriteLine("Derived Constructor");
}
public override void OverRideMe()
{
Console.WriteLine("In Derived Class's OverRideMe");
Console.Read();
}
public void NewMethod()
{
Console.WriteLine("In Derived Class's NewMethod");
Console.Read();
}
}
Main()的CIL(通用中间语言)代码如下所示:
.method private hidebysig static
void Main (
string[] args
) cil managed
{
// Method begins at RVA 0x2050
// Code size 42 (0x2a)
.maxstack 1
.entrypoint
.locals init (
[0] class ConsoleApplication1.Animal animal,
[1] class ConsoleApplication1.Dog dog
)
IL_0000: nop
IL_0001: newobj instance void ConsoleApplication1.Dog::.ctor()
IL_0006: stloc.0
IL_0007: ldloc.0
IL_0008: callvirt instance void ConsoleApplication1.Animal::OverRideMe()
IL_000d: nop
IL_000e: ldloc.0
IL_000f: castclass ConsoleApplication1.Dog
IL_0014: stloc.1
IL_0015: ldloc.1
IL_0016: callvirt instance void ConsoleApplication1.Animal::OverRideMe()
IL_001b: nop
IL_001c: ldloc.1
IL_001d: callvirt instance void ConsoleApplication1.Dog::NewMethod()
IL_0022: nop
IL_0023: call int32 [mscorlib]System.Console::Read()
IL_0028: pop
IL_0029: ret
} // end of method Program::Main
CIL中令我困扰的是:
IL_000f: castclass ConsoleApplication1.Dog
IL_0014: stloc.1
IL_0015: ldloc.1
IL_0016: callvirt instance void ConsoleApplication1.Animal::OverRideMe()
IL_001b: nop
IL_001c: ldloc.1
IL_001d: callvirt instance void ConsoleApplication1.Dog::NewMethod()
动物的 castclass 到 Dog 后,代码执行 dog.OverRideMe(); 。这被翻译为CIL
IL_0016:callvirt实例无效 ConsoleApplication1.Animal :: OverRideMe()
我已将动物对象转换为 Dog 类型。为什么要将 dog.OverRideMe(); 翻译成CIL中的上述语句?上面代码的输出是:
此输出与基类动物无关,但CIL仍然会调用它。
答案 0 :(得分:4)
您正在调用虚拟方法。虚方法调用由对象的运行时类型确定。您可以将它称为Dog
,但编译器仍会发出指令以确定在运行时调用的适当方法。从编译时类型的dog
开始,它遍历继承链,直到找到{{1}的“顶级”定义 1 并为它发出虚拟方法调用。在这种情况下,定义OverRideMe
的继承链中的最高位置是OverRideMe
;因此,它会为Animal
发出虚拟方法调用。
这是一个previous answer,可以帮助您了解更好的情况。
1 :定义方法的继承链中的最高位置。 必须在此处理解一些,以了解方法隐藏方式以及不会对此产生什么影响。
答案 1 :(得分:0)
它表示“callvirt” - 虚拟表与“Animal”类相关联,因此应该放置调用。在解析虚拟表之后,在运行时期间,将调用预期的方法。