我们首先来看一段代码: 1 class Program 2 { 3 static void Main(string[] args) 4 { 5 Father father=new Son(); 6 father.SayHi(); 7 Console.ReadKey(); 8 } 9 } 10 class Father 11 { 12 public void SayHi() 13 { 14 Console.WriteLine("我是父类的SayHi方法~!"); 15 } 16 } 17 class Son : Father 18 { 19 public void SayHi() 20 { 21 Console.WriteLine("我是子类的SayHi方法~!"); 22 } 23 }再来看一下该段代码的运行结果: 这时不禁有人就会问:我们new出来的是子类对象,为什么调用的是父类的方法呢??? 这就要从CLR的调用机制来解答了: 首先每个类型(Father,Son)都有自己的方法表,CLR在调用一个方法时,会判断这个方法是不是虚方法(下文将讲解虚方法的情况).如果不是虚方法,那么就去检查这个变量类型的方法表,找到了这个方法就执行它. 小结: 当父类对象指向子类实例时,即 Father father=new Son()时,父类和子类都存在SayHi方法(父类SayHi是非虚方法)时,当父类调用该方法时,即father.SayHi()时,调用的是父类的SayHi方法. 下面,我们来讲解父类中的方法是虚方法的情况,先看以下代码: 1 class Program 2 { 3 static void Main(string[] args) 4 { 5 Father father = new Son(); 6 father.SayHi(); 7 Console.ReadKey(); 8 } 9 } 10 class Father 11 { 12 public virtual void SayHi() 13 { 14 Console.WriteLine("我是父类的SayHi方法~!"); 15 } 16 } 17 class Son : Father 18 { 19 public override void SayHi() 20 { 21 Console.WriteLine("我是子类的SayHi方法~!"); 22 } 23 } 上面的代码运行结果是: 由此可见: 当父类调用的是虚方法,那么CLR会根据引用找到堆上的那个对象,根据对象的type pointer找到对象的真正类型(即GetType方法的返回类型),再调用该类型对应的方法,即子类的SayHi方法.