• C# 方法 虚方法的调用浅谈 引用kdalan的博文


    我们在面试中经常碰到有关多态的问题,之前我也一直被此类问题所困扰,闹不清到底执行哪个方法。

    先给出一道简单的面试题,大家猜猜看,输出是?

    复制代码
        public class A
        {
            public void MethodF() 
            { 
                Console.WriteLine("A.F"); 
            }
            public virtual void MethodG() 
            { 
                Console.WriteLine("A.G"); 
            }
        }
        public class B : A
        {
            new public void MethodF() 
            { 
                Console.WriteLine("B.F"); 
            }
            public override void MethodG() 
            { 
                Console.WriteLine("B.G"); 
            }
        }
        class Test
        {
            static void Main()
            {
                B b;
                b = new B();
                A a = b;
                a.MethodF();
                b.MethodF();
                a.MethodG();
                b.MethodG();
            }
    复制代码

    首先看一下虚方法的定义(MSDN):

      若一个实例方法的声明中含有 virtual 修饰符,则称该方法为虚拟方法。若其中没有 virtual 修饰符,则称该方法为非虚拟方法。

    以上面题目Test类Main中代码为例,简单说一下CLR创建对象的过程都做了什么事情

    1) 首先,声明一个引用类型变量 b,它仅是一个引用,保存在线程的栈上,用于将来存放B对象的有效地址。此时 b 未指向任何有效的实例,值为null,相关代码为:

         B b;

    2) 接下来,通过new执行对象的创建,即:

         b = new B();
    对象的实例保存在托管堆上,CLR在创建一个新对象的同时,还会创建它的类型对象(如果类型对象不存在)。

      对象实例在堆中的内存包括实例字段、类型对象指针、同步索引块,类型对象指针指向类型对象。

      类型对象在堆中分配的内存包括类型对象指针、同步索引块、静态字段、方法表。

    3)  A a = b;  这行代码首先声明一个类型为A的引用类型变量a,并将其实际地址指向b所指向的对象实例。

    4)  之后就是方法的调用,下面详细说一下C#中方法的调用:

        a.MethodF();

    当调用一个对象的方法时,会直接检查这个对象变量(a)的类型 ,找到堆中的类型对象,查看是否有该方法,没有则通过类型对象的类型对象指针向上回溯查找,直至找到,然后检查该方法是否为虚方法,如果非虚,直接调用,由于MethodF 方法是非虚的,因此直接调用输出A.F。

        a.MethodG();

    如果该方法为虚方法,即有virtual 关键字,则根据对象变量(a),去找到对象的实例类B,查找该类型对象中是否重新实现过该虚方法(override 关键字),如果有,OK执行,如果没有,向上检查其父类,直至找到然后执行,MethodG为虚方法,则会查找实例B,由于B中重写了MethodG,因此此处输出B.G


      通过上面的描述,开始的那道面试题,我们应该轻松可以得出输出,此处就不啰嗦了。 

      一般考多态的面试题中 virtual new override  几个关键字经常出现,new 关键字实现一个新的方法,同时隐藏基类的同名方法。 

  • 相关阅读:
    十六、Redis与MySQL数据双写一致性 —— Canal Demo
    十三、redis分布式锁:缓存续命
    十二、redis分布式锁:单机和多机及案例
    十七、缓存双写一致性之更新策略探讨
    十四、Redis的缓存过期淘汰策略
    自定义类型转换器
    [译]. NET 6 新增API 上
    EF 6 新特性三
    EF 6 新特性二
    [译]. NET 6 新增API 下
  • 原文地址:https://www.cnblogs.com/yzl495/p/4117617.html
Copyright © 2020-2023  润新知