• Emit学习(2)


    上一篇的介绍中, 并没有介绍到对象的创建过程, 这一篇主要就介绍一下, 对象的创建过程.

    其实熟悉了IL语法之后, 完全可以用Reflector反编译代码去查看. 而且正因为有这个工具, 可以对照着Reflecotr中的IL代码去写Emit的IL. 

    好了, 开始正题了, 还是从实例开始:

    一、示例

    首先建类(Person, Contacts):

    public class Person
        {
            public string Name { get; set; }
    
            public int Age { get; set; }
    
            public Contacts Contact { get; set; }
    
            public override string ToString()
            {
                return  string.Format("Name:{0}
    Age:{1}
    Contact:{2}", this.Name, this.Age, this.Contact.ToString());
            }
        }
        
        public class Contacts
        {
            public string Address { get; set; }
    
            public string Phone { get; set; }
    
            public string QQ { get; set; }
    
            public override string ToString()
            {
                var res = string.Format(@"[Address:{0}, Phone:{1}, QQ:{2}]", this.Address, this.Phone, this.QQ);
                return res;
            }
        }

    然后就可以写创建方法了:

    static void PersonTest()
    {
           var contacts = new Contacts {  Address="HeFei", Phone="15112341234", QQ="66666666"};
           var person = new Person { Name = "Wubi", Age = 20, Contact = contacts };
           Console.WriteLine(person.ToString());
     }

    反编译后的IL代码如下:

    .method private hidebysig static void PersonTest() cil managed
    {
        .maxstack 2
        .locals init (
            [0] class ConsoleApplication1.Contacts contacts,
            [1] class ConsoleApplication1.Person person,
            [2] class ConsoleApplication1.Contacts contacts2,
            [3] class ConsoleApplication1.Person person2)
    
        L_0000: nop 
        L_0001: newobj instance void ConsoleApplication1.Contacts::.ctor()
        L_0006: stloc.2 
    
        L_0007: ldloc.2 
        L_0008: ldstr "HeFei"
        L_000d: callvirt instance void ConsoleApplication1.Contacts::set_Address(string)  //contacts2.Address = "HeFei"
    
        L_0012: nop 
        L_0013: ldloc.2 
        L_0014: ldstr "15112341234"
        L_0019: callvirt instance void ConsoleApplication1.Contacts::set_Phone(string)
    
        L_001e: nop 
        L_001f: ldloc.2 
        L_0020: ldstr "66666666"
        L_0025: callvirt instance void ConsoleApplication1.Contacts::set_QQ(string)
    
        L_002a: nop 
        L_002b: ldloc.2 
        L_002c: stloc.0   //contacts = contacts2
    
        L_002d: newobj instance void ConsoleApplication1.Person::.ctor()
        L_0032: stloc.3 
        L_0033: ldloc.3 
        L_0034: ldstr "Wubi"
        L_0039: callvirt instance void ConsoleApplication1.Person::set_Name(string)
    
        L_003e: nop 
        L_003f: ldloc.3 
        L_0040: ldc.i4.s 20
        L_0042: callvirt instance void ConsoleApplication1.Person::set_Age(int32)
    
        L_0047: nop 
        L_0048: ldloc.3 
        L_0049: ldloc.0 
        L_004a: callvirt instance void ConsoleApplication1.Person::set_Contact(class ConsoleApplication1.Contacts)
    
        L_004f: nop 
        L_0050: ldloc.3 
        L_0051: stloc.1 
        L_0052: ldloc.1 
    
        L_0053: callvirt instance string [mscorlib]System.Object::ToString()
        L_0058: call void [mscorlib]System.Console::WriteLine(string)
        L_005d: nop 
        L_005e: ret 
    }

    二、补充的点

    可能有人注意到了, 此处调用方法时, 用的并不是Call, 而是 Callvirt, 什么时候用Call, 什么时候用Callvirt呢?

    首先, 我们看callvirt出现的位置, 是出现在public string Name{get;set;}中的, get, set其实是两个方法, 大家都知道的, 所以在给属性赋值时, 其实是调用的方法, 所以这里赋值并不是用的stloc之类的

    .method public hidebysig specialname instance void set_Name(string 'value') cil managed
    {
        .custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor()
        .maxstack 8
        L_0000: ldarg.0    //非静态方法中, arg0指的是this
        L_0001: ldarg.1    //arg1才是这里的'value'
        L_0002: stfld string ConsoleApplication1.Person::<Name>k__BackingField
        L_0007: ret 
    }

    其次, 我们先从字面的意思去看着两个指令

    Call 都知道是调用的意思, 那么Callvirt什么意思呢? 从前面的经验, 我想到, 这些指令并不是随便取的, 短到一个字母都是有特定意义的, 所以, 把Callvirt拆开来, 只看virt, 是不是有些熟悉的赶脚, 跟virtual这个单词比较一下, 就能明白Callvirt主要是调用那些方法了.

    嘿嘿, 没有再次了哦, 下面是call和callvirt的一些区别:

    1.call可以调用静态方法, 实例方法, 以及虚方法; 而callvirt只能调用实例方法和虚方法, 对于静态方法, 是心有余而力不足的

    2.call一般是以非虚的方法来调用函数的, 而callvirt是以多态的方式来调用函数的.

    至于异同的实例, 我就不给了, 大家可以看一下 : http://www.cnblogs.com/wang_yb/archive/2011/06/28/2092327.html

    还有一个哥们也写过call与callvirt的区别 : http://www.cnblogs.com/yingql/archive/2009/03/23/1420000.html

    网络时代就是好啊, 可以很容易的就能获取到别人的成果, 谢谢这些哥们了

  • 相关阅读:
    Android布局控件
    XAMPP里tomcat启动报错:Make sure you have Java JDK or JRE installed and the required ports are free
    poj 北京大学 2014研究生推免上机考试(校内)
    2014北大计算机学科保研直博夏令营上机poj考试
    《C++Primer》第四版学习笔记--持续更新中
    poj1986:Distance Queries
    poj2533:最长上升子序列
    poj1062:昂贵的聘礼
    黑书贪心例题之钓鱼 poj1042:Gone Fishing
    转:最小没出现的整数
  • 原文地址:https://www.cnblogs.com/elvinle/p/6005663.html
Copyright © 2020-2023  润新知