MSDN上对virtual方法的解释:试着翻译如下

当一个方法声明包含virtual修饰符,这个方法就是虚方法。如果没有virtual修饰符,那么就不是虚方法。

非虚方法的实现是不变的:不管该方法是被声明该方法的类的实例调用,还是该类的子类所调用,实现的结果都是一样。相比之下,虚方法会在子类中被取代。取代继承的虚方法的过程就是override。

在一个虚方法的调用中,方法所在的实例的运行时类型决定了实际哪个方法要被实现。在一个非虚方法的调用中,实例的编译时类型(编译时类型与运行时类型的区别)是决定性因素。准确地说,当一个参数列表为A,名叫N的方法在编译时类型C和运行时类型R的实例上调用时(R或者是C或者是C的子类),调用过程如下进行:

  • 首先,重载决议应用在C,N和A上,去从方法集合中选择一个在C中声明并继承的特定的方法M。
  • 然后,如果M是非虚方法,M就被调用。
  • 否则,M是虚方法,R中M派生程度最大的方法的实现被调用。

对于声明或继承于一个类的每个虚方法,相对于那个类都有一个派生程度最大的实现。对于每一个类R,派生程度最大的虚方法实现如下定义:

  • 如果R包含M的virtual声明,那么这就是M的派生程度最大的实现。
  • 否则,如果R包含M的override,那么,这就是M的派生程度最大的实现。
  • 否则,R的M派生程度最大的实现与R的直接父类的M最大派生程度最大实现相同。
using System;
class A
{
public void F() { Console.WriteLine("A.F"); }
public virtual void G() { Console.WriteLine("A.G"); }
}
class B: A
{
new public void F() { Console.WriteLine("B.F"); }
public override void G() { Console.WriteLine("B.G"); }
}
class Test
{
static void Main() {
B b = new B();
A a = b;
a.F();
b.F();
a.G();
b.G();
}
}

结果:

A.F
B.F
B.G
B.G

这么一大段,实在很难看懂。最后,根据《你必须知道的.NET》一书的对继承本质的描述,感觉清晰了一些。试着归纳如下:

对于给定的两个类:

   class A
{
public virtual void F()
{
Console.WriteLine("A.F");
}
}
class B :A
{
public override void F()
{
Console.WriteLine("B.F");
}
}

以及程序:

A a=new B();
a.F();//"B.F"

到底发生了什么呢?

根据下图:

显然A是编译时类型,指向了作为运行时类型B的实例。

当new B()时,首先将B的父类A所有的虚方法都复制了一份,然后和B中自己所有的方法对比,如果B中有override的虚方法,则以子类的方法替换父类的方法,同时添加子类其他的新方法。

因此,结果是B中的方法被执行。

另一个例子:

    class Z
{
public virtual void F1()
{
Console.WriteLine("Z.F");
}
}
class E:Z
{
public string type = "eType";
public override void F1()
{
Console.WriteLine(type);
}
}
class F : E
{
public string type = "fType";
public override void F1()
{
Console.WriteLine(type);
}
}
E e=new F();
Console.WriteLine(e.type);
e.F1();

结果出我意外:e.type里是eType,而e.F1()则是fType。

后者不难理解,因为是override,而前者是因为其为E类型,会首先访问离E类型创建最近的字段或方法。

最后,仍旧是MSDN上的一个例子

 class A
{
public virtual void F() { Console.WriteLine("A.F"); }
}
class B : A
{
public override void F() { Console.WriteLine("B.F"); }
}
class C : B
{
new public virtual void F() { Console.WriteLine("C.F"); }
}
class D : C
{
public override void F() { Console.WriteLine("D.F"); }
}
            D d = new D();
A a = d;
B b = d;
C c = d;
a.F();
b.F();
c.F();
d.F();

结果依次是:

B.F

B.F

D.F

D.F

两个D.F的结果好理解,为什么前两个是B.F呢?

因为d中其实有两个方法,一个是被隐藏的F(),其实现是B.F,另一个是显式的F(),其实现是D.F。根据”new则看类型,override只管新“,因为类型为A和B,那么其结果就为被隐藏的B.F。

virtual关键字的本质是什么?的更多相关文章

  1. C#基础知识五之abstract virtual关键字

    abstract 用关键字abstract修饰的类叫做抽象类,且只能作为基类,也不能实例化. 用abstract定义的抽象类中不一定只包含抽象方法 ,可以包含非抽象方法. abstract定义的方法一 ...

  2. virtual 关键字

    virtual 关键字用于修饰方法.属性.索引器或事件声明,并且允许在派生类中重写这些对象.例如,此方法可被任何继承它的类重写. public virtual double Area() { retu ...

  3. 浅析C#中new、override、virtual关键字的区别

    Virtual : virtual 关键字用于修饰方法.属性.索引器或事件声明,并使它们可以在派生类中被重写. 默认情况下,方法是非虚拟的.不能重写非虚方法. virtual 修饰符不能与 stati ...

  4. 浅析c++中virtual关键字

    http://blog.csdn.net/djh512/article/details/8973606 1.virtual关键字主要是什么作用? c++中的函数调用默认不适用动态绑定.要触发动态绑定, ...

  5. C++ 类的多态一(virtual关键字--构造函数深刻理解)

    //virtual关键字--构造函数深刻理解 #include<iostream> using namespace std; /* C语言编译器,c++编译器全部是静态链编,就是一段一段代 ...

  6. C++ 学习笔记 (七)继承与多态 virtual关键字的使用场景

    在上一篇 C++ 学习笔记 (六) 继承- 子类与父类有同名函数,变量 中说了当父类子类有同名函数时在外部调用时如果不加父类名则会默认调用子类的函数.C++有函数重写的功能需要添加virtual关键字 ...

  7. C++ 多态、虚函数(virtual 关键字)、静态联编、动态联编

    函数重写:(在子类中重写父类中的函数) 父类中被重写的函数  依然会继承  给子类. 子类中重写的函数将覆盖父类中的函数. 通过作用域分辨符  ::  可以访问到父类中的函数. 例如: #includ ...

  8. 还原virtual函数的本质-----C++

    当你每次看到C++类中声明一个virtual函数,特别是看到了一个virtual的虚构函数.你知道它的意思吗?你肯定会毫不犹豫的回答:不就是多态么...在运行时确定具体的行为么...完全正确,但这里我 ...

  9. virtual关键字

    出于多态的考虑,为了覆盖, 子类同名覆盖函数(函数名.参数.返回值都相同) virtual void print(): 这也许会使人联想到函数的重载,但稍加对比就会发现两者是完全不同的:(1)重载的几 ...

随机推荐

  1. Shell编程基础教程5--文本过滤、正则表达式、相关命令

    5.文本过滤.正则表达式.相关命令    5.1.正则表达式(什么是正则表达式?正则表达式怎么进行匹配?常用命令)        简介:            一种用来描述文本模式的特殊语法      ...

  2. scrollTop和scrollLeft的兼容解决万全方法

    1.各浏览器下 scrollTop的差异 IE6/7/8: 对于没有doctype声明的页面里可以使用  document.body.scrollTop 来获取 scrollTop高度 : 对于有do ...

  3. oracle使用dbms_metadata包取得所有对象DDL语句

    当我们想要查看某个表或者是表空间的DDL的时候,可以利用dbms_metadata.get_ddl这个包来查看. dbms_metadata包中的get_ddl函数详细参数 GET_DDL函数返回创建 ...

  4. Linux重置root密码步骤

    1.开机时任意按一个方向键,进入界面,选择linux系统,按e键进入2.然后用上下键选择kerner(内核)那一行,按e键进入编辑界面,编辑界面最后一行显示如下:(grub edit> kern ...

  5. Oracle备份及备份策略

    第二章. 了解备份的重要性 可以说,从计算机系统出世的那天起,就有了备份这个概念,计算机以其强大的速度处理能力,取代了很多人为的工作,但是,往往很多时候,它又是那么弱不禁风,主板上的芯片.主板电路.内 ...

  6. Linux下常用命令

    1.判断桌面环境是Gnome还是KDE #update-alternatives --display x-session-manager

  7. loadrunner中切割字符串

    下面函数的作用: http://blog.csdn.net/hgj125073/article/details/8447605 通过-与: 字符切割字符串,即-与:字符已经被\0 字符取代 char  ...

  8. hdu 1087 Super Jumping! Jumping! Jumping! 简单的dp

    Super Jumping! Jumping! Jumping! Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 ...

  9. 廖雪峰js教程笔记3

    JavaScript的函数在查找变量时从自身函数定义开始,从"内"向"外"查找.如果内部函数定义了与外部函数重名的变量,则内部函数的变量将"屏蔽&qu ...

  10. POJ 3349 HASH

    题目链接:http://poj.org/problem?id=3349 题意:你可能听说话世界上没有两片相同的雪花,我们定义一个雪花有6个瓣,如果存在有2个雪花相同[雪花是环形的,所以相同可以是旋转过 ...