面向对象--接口与抽象类的恩恩怨怨
接口与抽象类是面向对象编程中两个非常重要的角色,二者各自起着非常重要的作用。但是很多初学的朋友往往会对使用接口还是抽象类存在的很大的迷惑。就我自己的一点心得,发表一下拙见。
面向对象的一些回顾:
面向对象世界中有一个古老的法则:接口隔离原则,指的是不要把多个功能全部都集中在一个接口里面。接口实现的功能要相对单一;衍生开来可以得到另外一个结论:对一组或者称一系列功能的实现,尽量定义相对功能单一的小模块来实现这一组功能。这其实也是解耦和的体现。
那这跟我们的接口和抽象类有什么关系呢?那又得摆出另外一个法则:依赖倒置原则,针对接口编程而不是针对实现编程。
说到这,又会有一个新的问题蹦出来,这是自相矛盾啊,既然要针对接口编程还要抽象类干吗使?我们经常说面向对象,面向对象是来源于生活的。是人们要把对现实世界中的一系列方法论应用到程序设计当中来。从对象这一概念的引入我们就可以揣摩这一点。人类社会中有很多对象的概念,人、车、物体。不幸的是用程序来实现这些对象比在概念上定义对象要难很多。
(如果能达成这一共识,您可以继续往下看,否则就请看官您移步至留言讨论吧)
MS给出开发者的建议是,用抽象类来实现接口。子类再继承基类。
实例说明:
为什么要这么建议?OK,我们试着结合实际来说明一下这个问题吧。我们要造车。这个车有个基本的属性就是能移动、还必须有轮子。那我们就设计一个接口
1public interface ICar
2 {
3 string Wheel
4 {
5 get;
6 set;
7 }
8 void Move();
9 }
10
接下来的事情,就是实现了。造什么车都行,继承一下就行。随着科技的发展,我们的车想要飞了。此时当然不能修改这个接口,因为要遵循开闭原则。为什么要遵循?我们可以想一下,人坐上飞机能飞上天。但是也没见谁认为人有会飞这个特性的。那也好办,不许修改,那我再加一个接口。
1interface IFlyable
2 {
3 void Fly();
4 }
5
好,我们的飞行汽车最后应该是这样的。
1class FlyCar : ICar,IAerocraft
2 {
3 private string wheel = string.Empty;
4
5 public void Fly()
6 {
7 Console.WriteLine("{0}车飞起来了",this.wheel);
8 }
9 public string Engine
10 {
11 get
12 {
13 return wheel;
14 }
15 set
16 {
17 wheel = value;
18 }
19 }
20
21 public void Move()
22 {
23 Console.WriteLine("{0}轮车在走",this.wheel);
24 }
25 }
26
看起来很不错,车能飞能走了。那它现在他的祖宗到底车还是飞行器呢?我们自己在心里辩论一下吧。估计不是很容易辩清楚。
我们前面说过,面向对象的思想来源于现实生活。如果把这组例子引入到现实中来,造会飞的汽车。肯定是要在原有的汽车上面下功夫。比如你装上喷气动力装置,或者装上翅膀。
这只属于扩展功能,而不能说是继承基类。但上面的例子可以明显的看出,我们的飞行汽车已经成了杂交品种。分不出到底是车还是飞行器了。这里就可以知道为什么C#和JA V A都不支持多重继承基类了。避免杂交,减少耦合。
上面把车定义成接口并不完美,我们知道,一辆正常的车肯定能移动。这是天生的本质,不需要任何实现。但是上面还需要子类来实现这个功能。从这一点其实可以衍生出很多问题来。我们这里不做过多讨论。
重新设计这个系统。我们可以把移动,飞行都看成是一种行为。我们的车本身拥有Move 这个行为,是构成车基类的基本要素。
1interface IMoveable
2 {
3 void Move();
4 }
5 interface IFlyable
6 {
7 void Fly();
8 }
9public abstract class Car : IMoveable
10 {
11 public abstract string Wheel
12 {
13 get;
14 set;
15 }
16 public virtual void Move()
17 {
18 Console.WriteLine("车移动了");
19 }
20 }
21 public sealed class FlyCar : Car,IFlyable
22 {
23 private string wheel = string.Empty;
24 public override string Wheel
25 {
26 get
27 {
28 return wheel;
29 }
30 set
31 {
32 wheel = value;
33 }
34 }
35
36 public void Fly()
37 {
38 base.Move();
39 Console.WriteLine("汽车起飞成功!");
40 }
41 }
42 //在这里应用任何模式都很简单了
43 static void Main(string[] args)
44 {
45 FlyCar c = new FlyCar();
46 ((IFlyable)c).Fly();
47 ((Car)c).Move();
48 }
49
总结归纳:
其实类似的例子在我们的.NET Library里随处可见,例如Control类是继承于Component 和其他大量的接口的,而他们的基类却是MarshalByRefObject。因为他们归功到底又属于引用对象。
从上面的描述中,我们可以得出结论:
接口:
是某类行为或功能的抽象。是一种开关或者是契约。所以从字面上来理解就非常清楚了,西方神话中有很多和魔鬼定下契约来使自己的力量得到提升的故事。你必须定下这个契约才能得到你想要的力量。
抽象类:
对具体对象的最高抽象,这个对象拥有自己的最基本特征。
所以,从整体上来讲,抽象类和接口本质上都是是系统的最高抽象。从实际上来讲,二者抽象的对象不一样,就这一点导致了他们二者的应用的截然不同。
3句话:
1、接口抽象行为,抽象类抽象对象
2、用抽象类来实现接口。子类再继承基类
3、抽象类和接口本质上都是是系统的最高抽象
对象和接口(通俗讲解)
最近,还是看到很多人在问对象和接口的问题。
我原本以为,这已经不是个问题了,但是现在看来,它还困扰着很多程序员。
其实这个问题很简单,举例说明吧:
你是一个对象,继承自你的父母,你的gf也是一个对象,继承自她的父母,只有你和你的gf发生关系(比如打kiss)的时候,才会出现接口。这是接口的本质,接口就是一个对象要和另外一个对象发生关系时的协议。
很多误导人的书,把接口说成是为了解决多重继承问题,其实是作者不懂,而且还在误导别人。从下面的例子可以分析一下
type
TSun = class(TParent, ISomeInterface)
....
end;
持以上观点的人,只是从现象上看到,诶,这不是c++的多重继承吗?
错了,
上面的写法,真实的含义,应该是这样的,TSun,继承于TParent,说明TSun有TParent 大多数特性,还可能有自己的一些新特性,或者TParent没有完成的虚特性,TSun也完成了。这里主要解决代码复用问题。
但是,从语义上,却不能说TSun继承自ISomeInterface,ISomeInterface的出现说明这样一个问题:TSun,希望它的调用者按ISomeInterface的标准调用自己。
一切,都很清楚了,接口,和硬件的接口,协议,是一回事,跟面向对象没有任何关系,接口的出现,基于这样一个事实,对象间会发生关系,而接口具体明确的规定了发生关系的协议。就像一台电脑,本身是一个对象,但是至少又有电源,开关两个接口,人机交互又需要键盘和显示器两个接口。
所以,上面的例子是说,我遵守ISomeInterface接口,只要按ISomeInterface接口调用我,不会有任何问题。
从面向对象的角度看,public部分的东西,就是接口,只不过,interface的出现,更加强调了这个事实。它是大规模合作开发时的协议,是开发团队成员间必须遵守的规范。