深入Java核心Java中多态的实现机制_第1页
深入Java核心Java中多态的实现机制_第2页
深入Java核心Java中多态的实现机制_第3页
已阅读5页,还剩7页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、深入Java核心Java 中多态的实现机制1. 派生类对象的方法调用必须通过一个基类类 型的变量进行。2. 调用的方法必须在派生类中被定义。3. 调用的方法也必须被声明为基类的一个成 员。4. 基类和派生类中对应的方法的签名必须相 同。5. 基类和派生类的方法的返回对象类型必须相 同或者返回对象类型必须是协变的。6. 派生类的方法的访问说明符不能比基类有更 多的限制。深入 Java 核心 Java 中多态的实现机制多态性是面向对象程序设计代码重用的一个重要机制,我们曾不只一次的提到 Java 多态性。在 Java 运行时多态性:继承和接口的实现 一文中,我们曾详细介绍了 Java 实现运行时多

2、态性的动态方法调度; 今天我们再次深入 Java 核心,一起学习 Java 中多态性的实现。“ polymorphism( 多态)”一词来自希腊语,意为“多种形式”。多数Java 程序员把多态看作对象的一种能力,使其能调用正确的方法版本。尽管如此,这种面向实现的观点导致了多态的神奇功能,胜于仅 仅把多态看成纯粹的概念。Java 中的多态总是子类型的多态。 几乎是机械式产生了一些多态的行为, 使我们不去考虑其中涉及的 类型问题。本文研究了一种面向类型的对象观点,分析了如何将对象 能够表现的行为 和对象 即将表现的行 为分离开来。抛开 Java 中的多态都是来自继承的概念,我们仍然可以感到, Ja

3、va 中的接口是一组没有公 共代码的对象共享实现。多态的分类多态在面向对象语言中是个很普遍的概念.虽然我们经常把多态混为一谈,但实际上有四种不同类型的多态。在开始正式的子类型多态的细节讨论前,然我们先来看看普通面向对象中的多态。Luca Cardelli 和 Peter Wegner ("On Understanding Types, Data Abstraction, and Polymorphism"一文的作者,文章参考资源链接)把多态分为两大类-特定的和通用的-四小类:强制的,重载的,参数的和包含的。他们的结构如下:I輩制的I-持定前-III-莹载的多态一I|I-参数

4、的I-通用備-I卜-包含的在这样一个体系中,多态表现岀多种形式的能力。通用多态引用有相同结构类型的大量对象,他们有着共同的特征。特定的多态涉及的是小部分没有相同特征的对象。四种多态可做以下描述:强制的:一种隐式做类型转换的方法。重载的:将一个标志符用作多个意义。参数的:为不同类型的参数提供相同的操作。包含的:类包含关系的抽象操作。我将在讲述子类型多态前简单介绍一下这几种多态。强制的多态强制多态隐式的将参数按某种方法,转换成编译器认为正确的类型以避免错误。在以下的表达式中, 编译器必须决定二元运算符+'所应做的工作:2.0 + 2.02.0 + 22.0 + "2"第

5、一个表达式将两个 double 的 操作数相加; Java 中特别声明了这种用法。第二个表达式将 double 型和 int 相加。 Java 中没有明确定义这种运算。不过,编 译器隐式的将第二 个操作数转换为 double 型,并作 double 型的加法。做对程序员来说十分方便,否则将会抛出一个编译错 误,或者强制程序员显式的将 int 转换为 double 。第三个表达式将double与一个String相加。Java中同样没有定义这样的操作。 所以,编译器将double 转换成 String 类型,并将他们做串联。强制多态也会发生在方法调用中。假设类Derived继承了类Base,类C有

6、一个方法,原型为m(Base),在下面的代码中,编译器隐式的将Derived类的对象derived转化为Base类的对象。这种隐式的转换使m(Base)方法使用所有能转换成 Base类的所有参数。1. C c = new C();2.2. Derived derived =new Derived();4.5.c.m( derived );并且,隐式的强制转换,可以避免 类型转换的麻烦,减少编译错误。当然,编译器仍然会优先验证 符合定义的对象类型。重载的多态重载 允许用相同的运算符或方法,去表示截然不同的意义。 +'在上面的程序中有两个意思:两个 double 型的数相加;两个串相连。另

7、外还有整型相加,长整型,等等。这些运算符的重载,依赖于编译器 根据上下文做出的选择。以往的编译器会把操作数隐式转换为完全符合操作符的类型。虽然Java 明确支持重载,但 不支持用户定义的操作符重载。Java 支持用户定义的函数重载。一个类中可以有相同名字的方法,这些方法可以有不同的意义。这些 重载 的方法中,必须满足参数数目不同,相同位置上的参数类型不同。这些不同可以帮助编译器区分不 同版本的方法。编译器以这种唯一表示的特征来表示不同的方法,比用名字表示更为有效。据此,所有的多态行为都 能编译通过。强制和重载的多态都被分类为特定的多态,因为这些多态都是在特定的意义上的。这些被划入多态的 特性给

8、程序员带来了很大的方便。强制多态排除了麻烦的类型和编译错误。重载多态像一块糖,允许程序 员用相同的名字表示不同的方法,很方便。参数的多态参数多态 允许把许多类型抽象成单一的表示。例如, List 抽象类 中,描述了一组具有同样特征的对 象,提供了一个通用的模板。你可以通过指定一种类型以重用这个抽象类。这些参数可以是任何用户定义 的类型,大量的用户可以使用这个抽象类,因此参数多态毫无疑问的成为最强大的多态。乍一看,上面抽象类好像是 的功能。然而, Java 实际上并不支持真正的安全类型风 格的参数多态, 这也是 和 java.util 的其他集合类是用原始的 写的原 因(参考我的文章 "

9、;A Primordial Interface?" 以获得更多细节)。 Java 的单根继承方式解决了部分问题, 但没有发挥出参数多态的全部功能。 Eric Allen 有一篇精彩的文章“ Behold the Power of Parametric Polymorphism ”,描述了 Java 通用类型的需求, 并建议给 Sun 的 Java 规格需求 #000014 号文档 "Add Generic Types to the Java Programming Language."(参考资源链接)包含的多态包含多态通过值的类型和集合的包含关系实现了多态的行为

10、.在包括 Java 在内的众多面向对象语言 中,包含关系是子类型的。所以, Java 的包含多态是子类型的多态。在早期, Java 开发者们所提及的多态就特指子类型的多态。通过一种面向类型的观点,我们可以看到 子类型多态的强大功 能。以下的文章中我们将仔细探讨这个问题。为简明起见,下文中的多态均指包含 多态。面向类型观点5种类型,4个图1的UML类图给出了类和类型的简单继承关系,以便于解释多态机制。模型中包含一文中类和一个接口。虽然 UML中称为类图,我把它看成类型图。如"Thanks Type and Gentle Class,"所述,每个类和接口都是一种用户定义的类型。

11、按独立实现的观点(如面向类型的观点),下图中的每个矩形代表一种类型。从实现方法看,四种类型运用了类的结构,一种运用了接口的结构。+ni Q:Stririg rft2(s:String):String5interfacem2(s.Strio):stiiDQ.StringDerived |+iyi1 0:Strlfi +mO:String ISep dMtE+ml():9trlng+m2 (spring)'St ring+m30:strlng+m2(s: Siring): String+m4(J:Etf ng图1 :示范代码的UML类图以下的代码实现了每个用户定义的数据类型,我把实现写得很

12、简单。用这样的类型声明和类的定义,图2从概念的观点描述了 Java指令。Derived2 derived2 = new Derived2();ULI t图2 : Derived2对象上的引用上文中声明了 derived2 这个对象,它是 Derived2类的。图2种的最顶层把 Derived2引用描述成一 个集合的窗口,虽然其下的Derived2对象是可 见的。这里为每个 Derived2类型的操作留了一个孔。Derived2对象的每个操作都去映射适当的代码,按照上面的代码所描述的那样。例女口,Derived2对象映射了在Derived中定义的m1()方法。而且还重载了Base类的m1()方法

13、。一个Derived2的引用变量无权访问Base类中被重载的 m1()方法。但这并不意味着不可以用super.m1()的方法调用去使用这个方法。关系到derived2这个引用的变量,这个代码是不合适的。Derived2的其他的操作映射同样表明了每种类型 操作的代码执行。既然你有一个 Derived2对象,可以用任 何一个Derived2类型的变量去引用它。如图1所示,Derived, Base和IType都是Derived2的基类。所以,Base类的引用是很有用的。图3描述了以下语句的概念观点。Base base = derived2;Base虽然Base类的引用不用再访问 m3()和m4()

14、,但是却不会改变它 Derived2对象的任何特征及操作映射。无论是变量derived2还是base,其调用m1()或m2(String)所执行的代码都是一样的。两个引用之所以调用同一个行为,是因为 Derived2对象并不知道去调用哪个方法。对 象只知道什么 时候调用,它随着继承实现的顺序去执行。 这样的顺序决定了 Derived2对象调用Derived里的m1()方法, 并调用Derived2里的m2(String)方法。这种结果取决于对象本身的类型,而不是引用的类型。尽管如此,但不意味着你用 derived2和base引用的效果是完全一样的。如图 3所示,Base的引用 只能看到Base

15、类型拥有的操作。所以,虽然 Derived2有对方法m3()和m4()的映射,但是变量base不 能访问这些方法。运行期的Derived2对象保持了接受 m3()和m4()方法的能力。类型的限制使 Base的引用不能在编译 期调用这些方法。编译期的类型检查像一套铠甲,保证了运行期对象只能和正确的操作进行相互作用。换 句话说,类型定义了对象间相互作用的边界。多态的依附性类型的一致性是多态的核心。对象上的每一个引用,静态的类型检查器都要确认这样的依附和其对象的层次是一致的。当一个引用成功的依附于另一个不同的对象时,有趣的多态现象就产生了。(严格的说,对象类型是指类的定义。)你也可以把几个不同的引用

16、依附于同一个对象。在开始更有趣的场景前, 我们先来看一下下面的情况为什么不会产生多态。多个引用依附于一个对象图2和图3描述的例子是把两个及两个以上的引用依附于一个对象。虽然Derived2对象在被依附之后仍保持了变量的类型,但是,图3中的Base类型的引用依附之后,其功能减少了。结论很明显:把一个基类的引用依附于派生类的对象之上会减少其能力。一个开发这怎么会选择减少对象能力的方案呢?这种选择是间接的。假设有一个名为ref的引用依附于一个包含如下方法的类的对象:用一个Derived2的参数调用poly(Base)是符合参数类型检查的:方法调用把一个本地 Base类型的变量依附在一个引入的对象上。

17、所以,虽然这个方法只接受Base类型的参数,但 Derived2对象仍是允许的。开发这就不必选择丢失功能的方案。从人眼在通过Derived2对象时所看到的情况,Base类型引用的依附导致了功能的丧失。但从执行的观点看,每一个传入 polyl(Base)的参数都认为是Base的对象。执行机并不在乎有多个引用指向同一个对象,它只注重把指向另一个对象的引用传给方法。这些对象的类型不一致并不是主要问题。执行器只关心给运行时的对象找 到适当的实现。面向类型的观点展示了多态的巨大能力。附于多个对象的引用让我们来看一下发生在polyl(Base)中的多态行 为。下面的代码创建了三个对象,并通过引用传给pol

18、yl(Base):polyl(Base)的实现代码是调用传进来的参数的m1()方法。图3和图4展示了把三个类的对象传给方法时,面向类型的所使用的体系结构。BaseReferenceDerivedObjectBaseClassDerivedClassBaseCla$s图4:将Base引用指向Derived类,以及Base对象请注意每个图中方法 m1()的映射。图3中,m1()调用了 Derived类的代码;上面代码中的注释标明了ployl(Base)调用Derived.m1()。图4中Derived对象调用的仍然是 Derived类的m1()方法。最后,图 4 中,Base对象调用的m1()是B

19、ase类中定义的代码。多态的魅力何在?再来看一下polyl(Base)的代码,它可以接受任何属于Base类范畴的参数。然而,当他收到一个Derived2的对象时,它实际上却调用了Derived版本的方法。当你根据Base类派生出其他类时,如 Derived , Derived2 , poly1(Base) 都可以接受这些参数,并作出选择调用合适的方法。多态 允许你在完成 poly1(Base) 后扩 展它的用途。这看起来当然很神奇。基本的理解展示了多态的内部工作原理。在面向类型的观点中,底层的对象所实现的代码是非实质性的。 重要的是,类型检查器会在编译期间为每个引用选择合适的代码以实现其方法。

20、多态使开发者运用面向类型的观点,不考虑实现的细节。这样有助于把类型和实现分离(实际用处是把接口和实现分离)。对象接口多态依赖于类型和实现的分离,多用来把接口和实现分离。但下面的观点好像把Java 的关键字interface 搞得很糊涂。更为重要的使开发者们怎样理解短语“ the interface to an object",典型地,根据上下文,这个短语的意思是指一切对象类中所定义的方法,至一切对象公开的方法。这种倾向于以实现为中心的观点较 之于面向类型 的观点来说,使我们更加注重于对象在运行期的能力。图3中,引用面板的对象表面被标志成"Derived2 Object&qu

21、ot;。这个面板上列出了Derived2对象的所有可用的方法。但是要理解多态,我们必须从实现这一层次上解放出来,并注意面向类型的透视图中被标为 "Base Reference" 的面板。在这一层意思上,引用变量的类型指明了一个对象的表面。这只是一个表面,不是接口。在类型一致的原则下,我 们可以用面向类型 的观点,为一个对象依附多个引用。对 interface to an object 这个短语的理解没有 确定的理解。引用了面向类型观点的最大可能 如图 2 如图 3 所示。类型概念能使人获得把对在类型概念 中, the interface to an object refer

22、s的情形。把一个基类的引用指向相同的对象缩小了这样的观点 象间的 相互作用同实现细节分离的要领。相对于一个对象的接口,面向类型的观点更鼓励人们去使用一个对象的引用。引用类型规定了对象间的相互作用。当你考虑一个对象能做什么的时候,只需搞明白他的类型,而不需要去考虑他的实现细节。Java 接口以上所谈到的多态行为用到了类的继承关系所建立起来的子类型关系。Java接口同样支持用户定义的类型,相对地,Java的接口机制启动了建立在类型层次结构上的多态行为。假设一个名为ref的引用变量,并使其指向一个包含一下方法的类对象:为了弄明白poly2(IType)中的多态,以下的代码从不同的类创建两个对象,并分

23、另U把他们传给poly2(IType)上面的代码类似于关于polyl(Base)中的多态的讨论。poly2(IType)的实现代码是调用每个对象的本地版本的m3()方法。如同以前,代码的注释表明了每次调用所返回的CString类型的结果。图5表明了两次调用poly2(IType)的概念结构:图5:指向Derived2和Separate对象的IType引用方法poly1(Base)和poly2(IType)中所表现的多态行为的相似之处可以从透视图中直接看出来。把我 们在实现在一层上的理解再提高一层,就可以看到这两段代码的技巧。基类的引用指向了作为参数传进的类,并且按照类型的限制调用对象的方法。引用既不知道也不关心执行哪一段代码。编译期间的子类型关系检查保证了通过的对象有能力在被调用的时候选择合适的实现代码。然而,他们在实现层上有一个重要的差别。在poly1(Base)的例子中(图3和图4),Base-Derived-Derived2的类继承结构为子类型关系的建立提供了条件,并决定了方法去调用哪段代码。在poly2(IType)的例子中(如图 5),则是完全不同的动态发生的。Derived2和Separate不共享任何实现的层次,但 是他们还是通过IType的引用展示了多态的行为。这样的多态行为使 Java的接口的功能的重大意义显得很明显。图1中

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论