Emit学习-答疑篇-Call和Callvirt的区别

别等时光非礼了梦想. 提交于 2019-12-30 07:24:33

之前在Emit的学习过程中,多次碰到了方法的调用,发现有时候是使用Call而有时候是使用Callvirt,一直对这两者的区别不甚了解。然后就查阅了MSDN,MSDN中对这两者的解释为:

  l  Call:调用由传递的方法说明符指示的方法;

  l  Callvirt:对对象调用后期绑定方法,并且将返回值推送到计算堆栈上。

  但是看了之后还是很不明白,我想可能是因为中文版的缘故吧。今天下午再次看到了对Callvirt指令的解释,“对对象调用后期绑定方法”,突然想到,这个好像是指多态的意思吧?在一看virt,应该就是virtual的缩写,于是就更加肯定了自己的想法(外派在农行,不能上网,不然在园子随便一找就有结果了,伤心啊!),立马动手开始实践。

  我们用最经典的Animal的例子来验证这个想法,首先定义相关的类型,如下:

class Animal
{
    public virtual void Speak()
    {
        Console.WriteLine("Animal.Speak");
    }
}

class Cat : Animal
{
    public override void Speak()
    {
        Console.WriteLine("Cat.Speak");
    }
}

class Dog : Animal
{
    public override void Speak()
    {
        Console.WriteLine("Dog.Speak");
    }
}

 

  由于只是实现简单的方法调用,所以我们在这里选择使用DynamicMethod而不再创建动态程序集,顺便也可以演练下DynamicMethod的使用。要使用DynamicMethod我们首先要定义一个委托,用来执行方法的调用,定义如下:

private delegate void SpeakDelegate(Animal animal);

到时候我们通过此委托,传入一个Animal类或者其派生类的实例,并调用里面的Speak方法,从而验证之前的想法。由于方法的实现比较简单,这里就直接通过代码的注释进行讲解,代码如下:

class Program
{
    private delegate void SpeakDelegate(Animal animal);

    static void Main(string[] args)
    {
        //定义动态方法,没有返回值,传入参数为Animal,所在的模块选择为Program类所在的模块
        DynamicMethod dynamicSpeakWithCall = new DynamicMethod("DynamicSpeakWithCall", null, new Type[] { typeof(Animal) }, typeof(Program).Module);

        ILGenerator callIL = dynamicSpeakWithCall.GetILGenerator();

        //加载参数0 即 Animal类或其派生类的对象
        callIL.Emit(OpCodes.Ldarg_0);
        //通过Call指令调用Speak方法
        callIL.Emit(OpCodes.Call, typeof(Animal).GetMethod("Speak"));
        callIL.Emit(OpCodes.Ret);

        Console.WriteLine("SpeakWithCall:");
        SpeakDelegate SpeakWithCall = (SpeakDelegate)dynamicSpeakWithCall.CreateDelegate(typeof(SpeakDelegate));
        SpeakWithCall(new Animal());
        SpeakWithCall(new Cat());
        SpeakWithCall(new Dog());

        //定义动态方法,没有返回值,传入参数为Animal,所在的模块选择为Program类所在的模块
        DynamicMethod dynamicSpeakWithCallvirt = new DynamicMethod("DynamicSpeakWithCallvirt", null, new Type[] { typeof(Animal) }, typeof(Program).Module);

        ILGenerator callvirtIL = dynamicSpeakWithCallvirt.GetILGenerator();

        //加载参数0 即 Animal类或其派生类的对象
        callvirtIL.Emit(OpCodes.Ldarg_0);
        //通过Callvirt指令调用Speak方法
        callvirtIL.Emit(OpCodes.Callvirt, typeof(Animal).GetMethod("Speak"));
        callvirtIL.Emit(OpCodes.Ret);

        Console.WriteLine("SpeakWithCallvirt:");
        SpeakDelegate SpeakWithCallvirt = (SpeakDelegate)dynamicSpeakWithCallvirt.CreateDelegate(typeof(SpeakDelegate));
        SpeakWithCallvirt(new Animal());
        SpeakWithCallvirt(new Cat());
        SpeakWithCallvirt(new Dog());
    }
}

 

最后给出相应的输出结果:

SpeakWithCall:

 Animal.Speak

 Animal.Speak

 Animal.Speak

 SpeakWithCallvirt:

 Animal.Speak

 Cat.Speak

 Dog.Speak

 

  PS:由于学习Emit才只有几天的时间,所以上面的分析都显得有点肤浅,只是简单的记录下自己的学习过程,如果各位看官能够给我一点深层次的分析,我将不甚感激。

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!