`

接口和抽象类

阅读更多

接口和抽象类在设计层面上的区别

一个类一次只能继承一个抽象类,但可以实现若干个接口”。

首先注重这个要点的用词:“继承---类,实现---接口”,抽象父类和他的子类在体现出的是一种“继承”关系,要想使得“继承”关系合理,抽象父类和子类之间应该存在着“ is a“ 关系,即父类和子类在概念本质上应该是相同的。

而对于接口来说则不然,接口是拿来“实现”的,并不要求接口的实现类和接口定义在概念本质上是一致的,因为众所周知,接口只是定义了一组方法的框架,它的本质是行为特征的集合,规范。一个类实现了接口,仅仅是实现了接口所定义的规范而已,实现了接口的类并不“属于”接口!这时,应该说接口和实现了接口的子类的关系是一种“ like a” 的关系。

而且,一个类是可以实现多个接口的,在这种情况下,实现多个接口的类体现出的是“ like a like b like c” 的关系。在单继续的 Java 中,抽象类是做不到这点的。假设即使能这么做,假如一个类“ is a is b is c” ,这和我们对世界的理解也不太一致。因为我们看待事物时,习惯把事物归属一类,但同时可以具有其他类的特征。

 

考虑这样一个例子,假设我们在为一个生产厂家开发手机软件,问题领域中有一个关于手机的抽象概念,该手机具一些动作如开机,关机等,此时我们可以通过接口或者抽象类来定义一个表示该抽象概念的类型,定义方式分别如下所示:

使用接口方式定义手机:

Interface Mobile_phone{

Void pen () ;

Void close () ;

}

使用抽象类方式定义手机:

Abstract class Mobile_phone{

Abstract void open ();

Abstract void close ();

}

而具体的手机类型可以继承抽象类方式定义,或者使用实现接口的方式定义。看起来似乎使用接口和抽象类没有大的区别。

 

假如现在要求手机要集成有信用卡的功能。我们应该如何设计针对该例子的类结构呢?

信用卡的一些基本功能有,电子钱包,消费等,这些功能和手机的开机关机功能属于两个不同的概念,根据 ISP InterfaceSegregationPrinciple )原则应该把它们分别定义在代表这两个概念的抽象表示中。这时“信用卡”这个抽象概念的定义可能是这两种情况:

Interface Creditcard {

Void e_wallet () ; // 电子钱包

Void consume () ; // 消费

}

Abstract class Creditcard{

Abstract void e_wallet () ;

Abstract void consume ();

}

这时手机和银行卡这两个概念的定义方式就有了四种可能的组合,如下表:

手机

信用卡

 

 

方案 A

手机定义为抽象类

信用卡定义为抽象类

方案 B

手机定义为抽象类

信用卡定义为接口

方案 C

手机定义为接口

信用卡定义为抽象类

 

方案 D

手机定义为接口

信用卡定义为接口

方案 A 可以马上排除,因为 Java 不支持多继承, 手机要集成有信用卡 实现类这无法同时继续这两个抽象类。在这里可以看到抽象类在通过概念的组合来扩展功能的时候是不方便的。

 

研究一下方案 C ,如本文所述,这时实现这两个概念的类和这两个概念之间的关系就是:“ like” 手机,“ is” 信用卡。显然这和我们对问题领域的理解不符。因为“带信用卡功能的手机” 在概念本质上是手机,同时具有信用卡的功能 。所以这个方案是不合理的。除非我们是在为信用卡的制造商写软件,他们希望加入手机的功能。

而且,假如手机再扩展功能,如电子地图,导航器—--等等,把这种扩展的功能概念象“信用卡”一样定义为抽象类的话,由于 Java 的单继承机制,这是无法实现的。这道理和方案 A 一样,把用来扩展功能的概念定义为抽象类并不合适。

 

方案 B 应该是目前最合理的设计了,这时反映出的概念是“ is 手机, like 信用卡”。假如有扩展功能的话,可以再定义成接口,成为“ is 手机, like 信用卡 like 电子地图---”,从而正确的反应我们面对的问题域。

 

那方案 D 是不是就不行呢?

相对方案 C 来说,方案 D 的设计没有反映出“手机”是问题领域的本质的主体,使人有“到底我们在搞手机还是信用卡还是别的什么东西?”这个疑问。这个缺点是不容置疑的。但从另一方面来说,“手机”这个概念定义成接口,在软件规模扩大的前提下,也许为以后其他的组件的使用提供了方便。比如说,假设厂家又有一个“遥控器”的概念要我们设计,要把手机的功能设计进去,这时候“手机”假如是接口就方便了, implements 他就行。所以说,方案 D 是牺牲了概念的清楚性,得到了扩展性。

 

这里得到的结论就是:假如只是在定义一组行为框架的话,抽象类合适用来定义问题领域中的本质的抽象概念,接口合适用来定义扩展功能的抽象概念。

 

不能说这个结论就总结了抽象类和接口的全部区别,这只是问题的一小部分结论而已。再举一个例子来说明他们的另外一方面的区别。

 

在刚刚这个例子中,“手机”,“信用卡”仅仅是一组抽象方法,也就是概念中含有的只是行为框架没有实现,这时候定义成抽象类或接口都有自己的道理。假如概念中已经含有了实现,这时候就把应该概念定义成抽象类了。

 

比如一个” A 系列打印机”的抽象类,由他定义不同类型的打印机,那一系列的打印机打印页头,页脚的方案都是一样的,但打印页面主体比较复杂,各种具体型号的打印机的各有它们不同的打印方法,这时可以这么设计:

 

 

方案一:按照打印机应该打印完整页面的自然逻辑, body 抽象方法是打印机这个概念的一部分,设计为抽象类:

 

Abstract class A_SeriesPrinter{

Abstract protected void PrintBody () ;

Public void OutReport () {

PrintHeader () ;

PrintBody () ;

PrintFooter () ;

}

Protected void Draw Stringstr {/ 实现的代码 /}

Protected void PrintHeader () {Draw (“ Head“ ;/ 实现的代码 /}

Protected void PrintFooter () {Draw (“ Footer“ ;/ 实现的代码 /}

}

继续抽象类的代码:

Class XXPrinter extends A_SeriesPrinter{

Protected void PrintBody () {/ 实现的代码 /} // 这个一定要实现吧?

}

classYYPrinter extends A_SeriesPrinter{

protected void PrintBody () {/ 实现的代码 /}

}

运用的代码:

XXPrinterxx new XXPrinter () ;

Xx OutReport () ;

YYPrinteryy new YYPrinter () ;

Yy OutReport () ;

显然这个方案是简单而清楚的。

方案二:为了扩展性,硬把 body 抽象方法取出来成为一个接口,代码如下:

Abstract class A_SeriesPrinter{ // 思考一下,还用 abstract 么?

Protected void Draw Stringstr {/ 实现的代码 /}

Protected void PrintHeader () {Draw (“ Head“ ;/ 实现的代码 /}

Protected void PrintFooter () {Draw (“ Footer“ ;/ 实现的代码 /}

}

Interface Body{

void PrintBody () ;// 多了一个 Body 接口的概念

}

在这里先解决一个问题,假如 Printer 去掉了 PrintBody 抽象方法,都是实现了的方法,是不是就应该把它定义为普通的类呢?

答案是否定的,设计一个抽象概念为抽象类的意义,不是因为它含有抽象方法,而主要因为是他表示的概念不应该被实例化,即使它里头的方法全部是实现了的,想让子类继续的代码。在上面这个例子中,” A 系列打印机”这个概念,是不应该有实例的, 有实例的应该是具体型号的打印机 。所以,即便是全部是实现了的方法,方案二中的 A_SeriesPrinter 还是定义成抽象类更好。

 

好,继续看继续类并实现接口的代码:

Class XXPrinter extends A_SeriesPrinter implement Body {

Public void PrintBody () {;/ 实现的代码 /}

Public void OutReport () {//OutReport ()被迫移到了实现类

PrintHeader () ;

PrintBody () ;

PrintFooter () ;

}

}

Class YYPrinter extends A_SeriesPrinter implement sBody{

Public void PrintBody () {;/ 实现的代码 /}

Public void OutReport () {//OutReport ()被迫移到了实现类

PrintHeader () ;

PrintBody () ;

PrintFooter () ;

}

}

运用的代码:

XXPrinter xx new XXPrinter () ;

Xx OutReport () ;

YYPrinter yy new YYPrinter () ;

Yy OutReport () ;

这样做会显得很希奇和复杂: class XXPrinter extends Printer implements Body ?似 乎打印 Body 竟然是打印机的附加功能?还无故的多出了一个 Body 接口的概念。而且, OutReport ()被迫移到了各个实现类,代码变长而且复杂了。

所以这时抽象类是最好的选择。除非有业务要求需要把 Body 的打印从打印机分离出来。套到别的概念中去。这时才有考虑使它成为接口的可能,但再次提醒大家,代码会变得复杂。

追溯问题出现的源头,是因为 PrintBody ()这个抽象方法和打印机这个概念结合的太紧密了,它本身就是打印机功能的不可缺少的一部分。贪图接口语法上的灵活性,盲目的追求扩展性开放性,而不顾对问题领域的理解而建模,只要某一个概念中含有的行为框架都分离出来搞成接口,就会有一系列的编码上和理解上的麻烦,反而增加了代码的复杂性,自讨苦吃。

然而,即使在使用抽象类的场合,也不要忽视通过接口定义行为模型的原则。假如依靠于抽象类来定义行为,往往导致过于复杂的继承关系, 而通过接口定义行为能够更有效地分离行为与实现,为代码的维护和修改带来方便。

比如我扩展 A_SeriesPrinter 类,在打印后加个日志信息,如 viodoutLog ()方法什么的,那么我就不应该把它定义成 A_SeriesPrinter 类的抽象方法了,而是日志接口了。因为“日志”这概念不属于打印机的专有范畴。这样以后其他模块用到关于日志的操作规范时可以方便地用到这个日志接口。

所以,关键在于能否出色地结合业务要求对问题域进行理解分析。假如你没有做好这点,你就不能建立合理的模型。

 

总结:

1 接口能向上转型多个基本型别。

2 接口,接口让客户端程序员无法产生对象,确保这只是个接口,抽象类而无实体

3 你设计基本类 base class 时候,不带任何函数定义或成员变量时候,优先考虑 interface,( 如果必须带有函数定义或成员变量时候,改用 abstract class)

 

4 . 为了消除不变功能部分相同实现,避免代码冗余,应用抽象类

 

5. 多态实现中 , is- a like-a 关系

Is-a 是一种完全替换 base class 方法 (overring ) ,重载( overloader )功能,可以用 interface 来实现说接口和实现了接口的子类的关系是一种“ like a” 多态。

分享到:
评论

相关推荐

    java利用接口和抽象类改写求圆的面积和梯形的面积.docx

    ### Java利用接口和抽象类改写求圆的面积和梯形的面积 #### 深入理解Java的接口和抽象类 在Java编程语言中,接口(Interface)和抽象类(Abstract Class)都是用于实现多态性和代码复用的重要概念。它们都无法被...

    Java 接口和抽象类

    在上面的练习中,我们使用抽象类和接口来实现多态性。在 LivingThing 抽象类中,我们定义了一个抽象方法 dance(String dancingStyle),然后在 Human 和 Monkey 中实现了该方法。在 PersonInterface 接口中,我们定义...

    11.java接口和抽象类的区别.zip

    11.java接口和抽象类的区别.zip11.java接口和抽象类的区别.zip11.java接口和抽象类的区别.zip11.java接口和抽象类的区别.zip11.java接口和抽象类的区别.zip11.java接口和抽象类的区别.zip11.java接口和抽象类的区别....

    10.java接口和抽象类的相似性.zip

    10.java接口和抽象类的相似性.zip10.java接口和抽象类的相似性.zip10.java接口和抽象类的相似性.zip10.java接口和抽象类的相似性.zip10.java接口和抽象类的相似性.zip10.java接口和抽象类的相似性.zip10.java接口和...

    接口和抽象类使用详细实例源代码

    抽象(如抽象类和接口)作为契约,使得模块之间的交互基于稳定的抽象,而不是易变的具体实现。 在实际编程中,接口和抽象类的选择通常取决于特定场景的需求。如果需要为一组有共同行为的类提供基本实现,并且这些类...

    接口与抽象类区别

    那么,什么是抽象类和接口?它们之间有什么区别?下面,我们就来详细地探讨这个问题。 一、抽象类 抽象类是一种特殊的类,它不能被实例化,只能被继承。抽象类的主要特点是: 1. 抽象方法只作声明,而不包含实现...

    接口和抽象类的区别(面向对象)

    接口与抽象类的区别 抽象方法是必须实现的方法。就象动物都要呼吸。但是鱼用鳃呼吸,猪用肺呼吸。 动物类要有呼吸方法。怎么呼吸就是子类的事了。 现在有很多讨论和建议提倡用interface代替abstract类,两者从...

    Java 接口和抽象类区别

    在Java编程语言中,接口(Interface)与抽象类(Abstract Class)都是用于实现抽象和多态的重要工具。它们都允许我们定义行为的标准,但它们的使用场景、特点以及语法存在一定的差异。了解这些差异对于构建具有良好...

    1.5:接口和抽象类的区别.pdf

    4.实现抽象类和接口的类必须实现其中的所有方法。 抽象类中可以有非抽象方法。接口中则不能有实现方法。如果接口或者抽象类的子类不想实现则继续携程一个抽象方法 5.接口中定义的变量默认是public static final ...

    Java开发接口和抽象类的区别共2页.pdf.zip

    在Java编程语言中,接口(Interface)和抽象类(Abstract Class)都是用于实现多态性的关键概念,它们各自有特定的用途和特点。理解这两者的区别对于任何Java开发者来说都至关重要,因为正确地选择使用接口或抽象类...

    Java基础知识:接口和抽象类

    Java 基础知识:接口和抽象类 在 Java 编程中,接口和抽象类是两个重要的概念,它们都用于实现面向对象编程中的抽象和封装,但在使用方式和功能上有所不同。 1. 接口(Interface) 接口是一种抽象的数据类型,它...

    接口和抽象类的比较.pdf

    从代码层面进行比较,选择抽象类和接口的一个实际场景是:当多个类存在共同的行为或者属性时,我们可以把它们的共性抽象出来,形成一个抽象类。例如,在一个游戏角色的例子中,我们可以定义一个抽象类People,包含...

    接口和抽象类的区别精编版.doc

    接口和抽象类的区别精编版 在软件开发中,接口和抽象类是两种...抽象类和接口都是软件开发中重要的设计模式,它们之间的区别在于概念、方法实现和继承关系。正确地选择和使用抽象类和接口是软件开发中非常重要的一步。

    C#中的接口和抽象类

    通过“抽象类和接口练习”文件,你可以尝试创建不同的接口和抽象类,实现基本功能,并观察它们如何影响类的结构和交互。这将有助于巩固理论知识,并提高实战技能。在实践中,不断探索和比较接口与抽象类的使用场景,...

Global site tag (gtag.js) - Google Analytics