转载:http://blog.csdn.net/suncherrydream/article/details/8423991

若一个实例方法声明前带有virtual关键字,那么这个方法就是虚方法。 虚方法与非虚方法的最大不同是,虚方法的实现可以由派生类所取代,这种取代是通过方法的重写实现的(以后再讲) 虚方法的特点: 虚方法前不允许有static,abstract,或override修饰符 虚方法不能是私有的,因此不能使用private修饰符 虚方法的执行: 我们知道一般函数在编译时就静态地编译到了执行文件中,其相对地址在程序运行期间是不发生变化的, 而虚函数在编译期间是不被静态编译的,它的相对地址是不确定的,它会根据运行时期对象实例来动态判断要调用的函数, 其中那个申明时定义的类叫申明类,那个执行时实例化的类叫实例类。 如:A a =new B(); 其中A是申明类,B是实例类。 1.当调用一个对象的函数时,系统会直接去检查这个对象申明定义的类,即申明类,看所调用的函数是否为虚函数; 2.如果不是虚函数,那么它就直接执行该函数。而如果是一个虚函数,那么这个时候它就不会立刻执行该函数了,而是开始检查对象的实例类。 3.在这个实例类里,他会检查这个实例类的定义中是否有实现该虚函数或者重新实现该虚函数(通过override关键字)的方法, 如果有,它就不会再找了,而是马上执行该实例类中实现的虚函数的方法。而如果没有的话,系统就会不停地往上找实例类的父类, 并对父类重复刚才在实例类里的检查,直到找到第一个重载了该虚函数的父类为止,然后执行该父类里重载后的函数。 例1:

    class A
{
public virtual void Sum()
{
Console.WriteLine("I am A Class,I am virtual sum().");
}
}
class Program
{
static void Main(string[] args)
{
A a=new A(); // 定义一个a这个A类的对象.这个A就是a的申明类,实例化a对象,A是a的实例类  
a.Sum();
Console.Read();
}
}

执行a.Sum: 1.先检查申明类A 2.检查到是sum是虚拟方法 3.转去检查实例类A,结果是题本身 4.执行实例类A中实现Sum的方法 5.输出结果 I am A Class,I am virtual sum(). 例2:

class A
{
public virtual void Sum()
{
Console.WriteLine("I am A Class,I am virtual sum().");
}
}
class B : A
{
public override void Sum() // 重新实现了虚函数
{
Console.WriteLine("I am B Class,I am override sum().");
} }
class Program
{
static void Main(string[] args)
{
A a=new B(); // 定义一个a这个A类的对象.这个A就是a的申明类,实例化a对象,B是a的实例类  
a.Sum();
Console.Read();
}
}

执行a.Sum: 1.先检查申明类A 2.检查到是虚拟方法 3.转去检查实例类B,有重写的方法 4.执行实例类B中的方法 5.输出结果 I am B Class,I am override sum(). 例3:

    class A
{
public virtual void Sum()
{
Console.WriteLine("I am A Class,I am virtual sum().");
}
}
class B : A
{
public override void Sum() // 重新实现了虚函数
{
Console.WriteLine("I am B Class,I am override sum().");
} }
class C : B
{ }
class Program
{
static void Main(string[] args)
{
A a=new C();// 定义一个a这个A类的对象.这个A就是a的申明类,实例化a对象,C是a的实例类  
a.Sum();
Console.Read();
}
}

执行a.Sum: 1.先检查申明类A 2.检查到是虚拟方法 3.转去检查实例类C,无重写的方法 4.转去检查类C的父类B,有重写的方法 5.执行父类B中的Sum方法 6.输出结果 I am B Class,I am override sum().  例4:

   class A
{
public virtual void Sum()
{
Console.WriteLine("I am A Class,I am virtual sum().");
}
}
class B : A
{
public new void Sum() //覆盖父类里的同名函数,而不是重新实现
{
Console.WriteLine("I am B Class,I am new sum().");
} }
class Program
{
static void Main(string[] args)
{
A a=new B();
a.Sum();
Console.Read();
}
}

执行a.Sum: 1.先检查申明类A 2.检查到是虚拟方法 3.转去检查实例类B,无重写的(这个地方要注意了,虽然B里有实现Sum(),但没有使用override关键字,所以不会被认为是重写) 4.转去检查类B的父类A,就为本身 5.执行父类A中的Sum方法 6.输出结果 I am A Class,I am virtual sum().  那么如果在例4里,申明的是类B呢?

    class A
{
public virtual void Sum()
{
Console.WriteLine("I am A Class,I am virtual sum().");
}
}
class B : A
{
public new void Sum() //覆盖父类里的同名函数,而不是重新实现
{
Console.WriteLine("I am B Class,I am new sum().");
} }
class Program
{
static void Main(string[] args)
{
B b=new B();
b.Sum();
Console.Read();
}
}

执行B类里的Sum(),输出结果I am B Class,I am new sum(). 可以使用抽象函数重写基类中的虚函数吗? 答案是可以的。

    class A
{
public virtual void PrintFriends()
{
Console.WriteLine("A.PrintFriends()");
}
}
abstract class B : A
{
public abstract override void PrintFriends(); //使用override 修饰符,表示抽象重写了基类中该函数的实现
}
abstract class C : A
{
public abstract new void PrintFriends(); //使用 new 修饰符显式声明,表示隐藏了基类中该函数的实现
}

密封类可以有虚函数吗? 可以,基类中的虚函数将隐式的转化为非虚函数,但密封类本身不能再增加新的虚函数

    class A
{
public virtual void Fun()
{
Console.WriteLine("I am A.");
}
}
sealed class Program:A
{
public override void Fun()
{
Console.WriteLine("I am B.");
}
static void Main(string[] args)
{
Program p = new Program();
p.Fun();
Console.Read();
}
}

最新文章

  1. TypeScript 素描 - 接口
  2. bat命令总结
  3. php 函数汇总
  4. 教程和工具--用wxPython编写GUI程序的
  5. (八) 一起学 Unix 环境高级编程 (APUE) 之 信号
  6. 重新想象 Windows 8 Store Apps (65) - 后台任务: 音乐的后台播放和控制
  7. 推荐!Sublime Text 最佳插件列表
  8. 恢复Linux下被误删除的文件(笔记)
  9. linux动态库加载RPATH, RUNPATH
  10. HBase源代码分析之HRegion上MemStore的flsuh流程(一)
  11. 【转】详解JavaScript中的异常处理方法
  12. 分解数据表(将一个datatable按数据量分隔成多个table)
  13. Mac下的Chrome或Safari访问跨域设置,MBP上使用模拟器Simulator.app或iphone+Safari调试网页
  14. saltstack自动化运维系列⑥SaltStack实践安装配置HAproxy
  15. 您无法登陆系统。原因可能是您的用户记录或所属的业务部门在Microoft Dynamics CRM中已被禁用
  16. PHP ActiveRecord demo栗子中 关于类名 的问题
  17. OSWorkFlow 学习
  18. VS2010/MFC编程入门之五十二(Ribbon界面开发:创建Ribbon样式的应用程序框架)
  19. SimpleXML概述
  20. 半夜思考之查漏补缺 , Spring 中的 Bean 继承机制

热门文章

  1. DSAPI多功能组件编程应用-网络相关(上)
  2. ARM与FPGA通过spi通信设计2.spi master的实现
  3. Vue中父组件传子组件
  4. 自然底数e的意义是什么?
  5. jQuery省市区三级联动菜单
  6. c#高级编程_第10版 云盘地址
  7. C#中Windows窗体工具栏
  8. c/c++ 多线程 参数传递
  9. Java 集合系列(二)—— ArrayList
  10. SQLServer之创建辅助XML索引