一、前面的话
对于C#中的接口、抽象类中的抽象方法以及虚方法的比较,网上有很多的例子,讲得也很到位,这篇博文的目的一方面是为了巩固自己的基础知识,另一方面是能够让初学者更加深刻、轻松地去理解和接受这三者之间的区别与联系。
二、接口与抽象类
首先说说接口和抽象类的区别与联系,接口是什么?接口是一组规则,它对应着自然界中“如果你是……则必须能……”的思想,如:猫猫是动物,动物可以吃食物、可以移动,也就是说某个类实现了某个接口,就必须实现该接口的规则(方法),如果接口中的方法(规则)太多,可以考虑接口分离,接口是可以继承的,当然这不是本篇文章的重点。既然接口是一组规则,那么它的作用就是约束其他实现了该接口的类,而自己本身不能实例化。既然接口是一组规则,那么这个规则是什么样呢?就是包含了一系列的方法,在具体一点就是没有修饰符和方法体的方法。好了,现在在重新理解“接口是一组规则”这句话,接口就是包含了一系列没有修饰符和方法体的方法的集合,这就是接口。结合代码看一下:
1 interface IAnimal 2 { 3 //public void run(); 4 //The modifier 'public' is not valid for this item 5 void run(); 6 void eat(); 7 string speak(); 8 }
接口中的方法不能有修饰符和方法体,如果加上修饰符就会报错(第4行)。而实现了该接口的类必须实现其中的run、eat、speak方法。此外,接口中不能有字段。
1 class Cat:IAnimal 2 { 3 #region IAnimal Members 4 5 public void run() 6 { 7 Console.WriteLine("run run!"); 8 } 9 10 public void eat() 11 { 12 Console.WriteLine("delicious~"); 13 } 14 15 public string speak() 16 { 17 return "miao~"; 18 } 19 20 #endregion 21 }
Cat类实现了IAnimal接口,那么它必须实现IAnimal中的所有方法(规则)。
理解了接口,抽象类就比较简单了,与接口相同的是,抽象类的抽象方法不能有方法体,都是需要子类来实现具体的逻辑(如Cat中的eat方法)。但是,抽象类中可以有字段,抽象类其实就是一个特殊的普通类,只要有抽象方法,那么这个类就是抽象类。
1 abstract class AbstractCat 2 { 3 //field 4 private double weight; 5 //abstract method 6 public abstract void eat(); 7 //normal method 8 public void run() 9 { 10 Console.WriteLine("run run!"); 11 } 12 } 13 14 class BlackCat:AbstractCat 15 { 16 //Implement abstract method 17 public override void eat() 18 { 19 Console.WriteLine("delicious!"); 20 } 21 }
三、抽象方法与虚方法
我们可以把抽象方法看成没有方法体的虚方法,二者都可以被子类重写,并以override关键字修饰。不同的是,虚方法必须有方法体。
1 abstract class AbstractCat 2 { 3 //field 4 private double weight; 5 //abstract method 6 public abstract void eat(); 7 //normal method 8 public void run() 9 { 10 Console.WriteLine("run run!"); 11 } 12 } 13 14 class BlackCat:AbstractCat 15 { 16 //Implement abstract method 17 public override void eat() 18 { 19 Console.WriteLine("delicious!"); 20 } 21 22 public virtual string speak() 23 { 24 return "Miao!"; 25 } 26 27 //public virtual void run(); 28 } 29 30 class ChinaBlackCat : BlackCat 31 { 32 public override string speak() 33 { 34 return base.speak(); 35 } 36 }
四、细说虚方法
虚方法也就是虚拟函数,我们习惯称之为虚函数,虚函数从C#的程序编译的角度来看,它和其它一般的函数有什么区别呢?一般函数在编译时就静态地编译到了可执行文件中,其相对地址在程序运行期间是不发生变化的,也就是写死了的!而虚函数在编译期间是不被静态编译的,它的相对地址是不确定的,它会根据运行时期对象实例来动态判断要调用的函数,其中那个声明时定义的类叫声明类,执行时实例化的类是实例类。虚函数被调用时会执行下列检查:
1、当调用一个对象的函数时,系统会直接去检查这个对象申明定义的类,即申明类,看所调用的函数是否为虚函数;
2、如果不是虚函数,则直接执行该函数。而如果有virtual关键字,也就是一个虚函数,那么这个时候它就不会立刻执行该函数了,而是转去检查对象的实例类;
3、在这个实例类中,编译器会检查这个实例类的定义中是否有重写该虚函数(通过override关键字),如果有,那么编译器不会再继续寻找父类,而马上执行该实例类中的这个重新实现的函数。而如果没有的话,编译器就会不停地往上找实例类的父类,并对父类重复刚才在实例类里的检查,直到找到第一个重载了该虚函数的父类为止,然后执行该父类里重载后的函数。
理解上面的规则就不难判断如下代码的运行结果了:
1 namespace VirtualMethod 2 { 3 class Program 4 { 5 static void Main(string[] args) 6 { 7 A a; // 定义一个a这个A类的对象.这个A就是a的申明类 8 A b; // 定义一个b这个A类的对象.这个A就是b的申明类 9 A c; // 定义一个c这个A类的对象.这个A就是b的申明类 10 A d; // 定义一个d这个A类的对象.这个A就是b的申明类 11 12 a = new A(); // 实例化a对象,A是a的实例类 13 b = new B(); // 实例化b对象,B是b的实例类 14 c = new C(); // 实例化b对象,C是b的实例类 15 d = new D(); // 实例化b对象,D是b的实例类 16 17 a.Function(); // 执行a.Function:1.先检查申明类A 2.检查到是虚拟方法 3.转去检查实例类A,就为本身 4.执行实例类A中的方法 5.输出结果 Function In A 18 b.Function(); // 执行b.Function:1.先检查申明类A 2.检查到是虚拟方法 3.转去检查实例类B,有重载的 4.执行实例类B中的方法 5.输出结果 Function In B 19 c.Function(); // 执行c.Function:1.先检查申明类A 2.检查到是虚拟方法 3.转去检查实例类C,无重载的 4.转去检查类C的父类B,有重载的 5.执行父类B中的Function方法 5.输出结果 Function In B 20 d.Function(); // 执行d.Function:1.先检查申明类A 2.检查到是虚拟方法 3.转去检查实例类D,无重载的(这个地方要注意了,虽然D里有实现Function(),但没有使用override关键字,所以不会被认为是重载) 4.转去检查类D的父类A,就为本身 5.执行父类A中的Function方法 5.输出结果 Function In A 21 D d1 = new D(); 22 d1.Function(); // 执行D类里的Function(),输出结果 Function In D 23 Console.ReadLine(); 24 } 25 26 27 class A 28 { 29 public virtual void Function() // 注意virtual,表明这是一个虚拟函数 30 { 31 Console.WriteLine("Function In A"); 32 } 33 } 34 35 class B : A // 注意B是从A类继承,所以A是父类,B是子类 36 { 37 public override void Function() // 注意override ,表明重新实现了虚函数 38 { 39 Console.WriteLine("Function In B"); 40 } 41 } 42 43 class C : B // 注意C是从A类继承,所以B是父类,C是子类 44 { 45 } 46 47 class D : A // 注意B是从A类继承,所以A是父类,D是子类 48 { 49 public new void Function() // 注意new ,表明覆盖父类里的同名类 50 { 51 Console.WriteLine("Function In B"); 52 } 53 } 54 55 } 56 }