多态性:
- 是什么
- 怎么做
优点:
- 代码组织以及可读性均能获得改善
- 创建“易于扩展”的程序
Polymorphism - 动态绑定、推迟绑定或者运行期绑定
能够不顾衍生类,只让自己的代码与基础类打交道,那么省下的工作量将是难以估计的。
可在运行期间判断对象的类型,并分别调用适当的方法;
也就是说,编译器此时已然不知道对象的类型,但方法调用机制能自己去调查,找到正确的方法主体。
Java中绑定的所有方法都采用后期绑定技术,除非一个方法已被声明成final
final的使用:
- 防止其他人覆盖这个方法
- 有效“关闭”动态绑定,或者告诉编译器不需要进行动态绑定 - 编译器就可为final方法调用生成效率更高的代码
在一个设计良好的OOP程序中,而且只与基础类接口通信。
我们说这样的程序具有“扩展性”,因为可以从通用的基础类继承新的数据类型,从而新添一些功能。如果是为了适应新类的要求,那么对基础类接口进行操纵的方法根本不需要改变。
将发生改变的东西同没有发生改变的东西区分开
// accidentally changing the interface class NoteX{ public static final int MIDDLE_C = 0, C_SHARP=1, C_FLAT=2; } class InstrumentX{ public void play(int Notex){ System.out.println("InstrumentX.play()"); } } class WindX extends InstrumentX{ // OOPS! Changes the method interface: public void play(NoteX n){ System.out.println("WindX.play(NoteX n)"); } } public class WindError { public static void tune(InstrumentX i){ // ... i.play(NoteX.MIDDLE_C); } public static void main(String[] args) { WindX flute = new WindX(); tune(flute); //Not the desired behavior! } } /** * “过载”是指同一样东西在不同的地方具有多种含义; * “覆盖”是指它随时随地都只有一种含义,只是原先的含义完全被后来的含义取代了。 * 在tune 中,“InstrumentX i”会发出play()消息,同时将某个 NoteX 成员作为自变量使用(MIDDLE_C)。 * 由于NoteX 包含了int 定义,过载的play()方法的int 版本会得到调用。同时由于它尚未被“覆盖”,所以会使用基础类版本。 */
抽象类:
Java 专门提供了一种机制,名为“抽象方法”。它属于一种不完整的方法,只含有一个声
明,没有方法主体。
包含了抽象方法的一个类叫作“抽象类”。如果一个类里包含了一个或多个抽象方法,类就必须指定成
abstract(抽象)。否则,编译器会向我们报告一条出错消息。
编译器可保证抽象类的“纯洁性”,我们不必担心会误用它。
如果从一个抽象类继承,而且想生成新类型的一个对象,就必须为基础类中的所有抽象方法提供方法定义。如果不这样做(完全可以选择不做),则衍生类也会是抽象的,而且编译器会强迫我们用abstract 关键字标志那个类的“抽象”本质。
接口:
“interface”(接口)关键字使抽象的概念更深入了一层。
我们可将其想象为一个“纯”抽象类。它允许创建者规定一个类的基本形式:方法名、自变量列表以及返回型,但不规定方法主体。
接口也包含了基本数据类型的数据成员,但它们都默认为static 和final。
注意接口中的每个方法都严格地是一个声明,它是编译器唯一允许的。
接口只是比抽象类“更纯”的一种形式;
但它用途并不止于此,由于接口根本没有具体的实施细节 - 也就是说,没有与存储空间与“接口”关联在一起 - 所以没有任何办法可以防止多个接口合并到一起。
/** * 接口的规则是:我们可以从它继承(稍后就会看到),但这样得到的将是另一个接口。 * 如果想创建新类型的一个对象,它就必须是已提供所有定义的一个类。尽管Hero 没有为 fight()明确地提供 * 一个定义,但定义是随同ActionCharacter 来的,所以这个定义会自动提供,我们可以创建Hero 的对象。 * * ActionCharacter.fight() * ActionCharacter.fight() */ // Multiple interfaces interface CanFight{ void fight(); } interface CanSwim{ void swim(); } interface CanFly{ void fly(); } class ActionCharacter{ public void fight(){System.out.println("ActionCharacter.fight()");} } // 合并一个具体类与接口的时候,具体类必须首先出现,然后才是接口(否则编译器会报错) class Hero extends ActionCharacter implements CanFight, CanSwim, CanFly{ public void swim(){} public void fly(){} } public class Adventure { static void t(CanFight x){x.fight();} static void u(CanSwim x){x.swim();} static void v(CanFly x){x.fly();} static void w(ActionCharacter x){x.fight();} public static void main(String[] args) { Hero i = new Hero(); t(i); // Treat it as a CanFight u(i); // Treat it as a CanSwim v(i); // Treat it as a CanFly w(i); // Treat it as an ActionCharacter } }
接口可以:能上溯造型至多个基础类。
所以假如想创建的基础类没有任何方法定义或者成员变量,那么无论如何都愿意使用接口,而不要选择抽象类。事实上,如果事先知道某种东西会成为基础类,那么第一个选择就是把它变成一个接口。只有在必须使用方法定义或者成员变量的时候,才应考虑采用抽象类。
由于置入一个接口的所有字段都自动具有static 和final 属性,所以接口是对常数值进行分组的一个好工具:
// Using interfaces to create groups of constants public interface Months { int JANUARY = 1, FEBRUARY = 2, MARCH = 3, APRIL = 4, MAY = 5, JUNE = 6, JULY = 7, AUGUST = 8, SEPTEMBER = 9, OCTOBER = 10, NOVEMBER = 11, DECEMBER = 12; }
// Returning a handle to an inner class abstract class Contents{ abstract public int value(); } interface Destination{ String readLabel(); } public class Parcel3 { private class PContents extends Contents{ private int i = 11; public int value(){ return i; } } protected class PDestination implements Destination{ private String label; private PDestination(String whereTo){ label = whereTo; } public String readLabel(){ return label; } } public Destination dest(String s){ return new PDestination(s); } public Contents cont(){ return new PContents(); } } class Test{ public static void main(String[] args){ Parcel3 p = new Parcel3(); Contents c = p.cont(); Destination d = p.dest("Su Zhou"); //Illegal -- can't access private class: //! Parcel3.PContents c = p.new PContents(); } }
package chapter07.innerscopes; interface Destination { String readLabel(); } package chapter07.innerscopes; interface Contents { int value(); } package chapter07.innerscopes; public class Wrapping { private int i; public Wrapping(int x){ i = x; } public int value(){ return i; } } package chapter07.innerscopes; public class Parcel4 { public Destination dest(String s){ class PDestination implements Destination{ private String label; private PDestination(String whereTo){ label = whereTo; } public String readLabel(){ return label; } } return new PDestination(s); } public static void main(String[] args) { Parcel4 p = new Parcel4(); Destination d = p.dest("Su Zhou"); } } package chapter07.innerscopes; public class Parcel6 { public Contents cont(){ // 创建从Contents 衍生出来的匿名类的一个对象 // 由new表达式返回的句柄会自动上溯造型成一个Contents句柄。 return new Contents(){ private int i = 11; public int value(){ return i; } }; // Semicolon required in this case } /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub } } package chapter07.innerscopes; // 若试图定义一个匿名内部类,并想使用在匿名内部类外部定义的一个对象,则编译器要求外部对象为final属性。 public class Parcel8 { // Argument must be final to use inside anonymous inner class public Destination dest(final String dest){ return new Destination() { private String label = dest; @Override public String readLabel() { return label; } }; } public static void main(String[] args) { Parcel8 p = new Parcel8(); Destination d = p.dest("Su Zhou"); } }
// 必须利用外部类的一个对象生成内部类的一个对象: Parcel11.Contents c = p.new Contents(); // 除非已拥有外部类的一个对象,否则不可能创建内部类的一个对象
// 先是封装类的名字,再跟随一个$, // 再跟随内部类的名字。 WithInner$Inner.class // 如果内部类是匿名的,那么编译器会简单地生成数字,把它们作为内部类标识符使用
内部类:
- 在单独一个类里表达一个控制框架应用的全部实施细节,从而完整地封装与那个实施有关的所有东西。内部类用与表达多种不同类型的的action,用于解决实际的问题。
- 内部类使我们具体的实施变得更加巧妙,因为能方便地访问外部类的任何成员。
相关推荐
Get a grounding in polymorphism and other fundamental aspects of object-oriented program design and implementation, and learn a subset of design patterns that any practicing Java professional simply ...
#Java中的多态性和继承###是什么? 一个Java项目,以演示和理解Java中的多态性和继承。 该项目使用Java 7。 ###如何使用? 这是一个eclipse项目,因此要从eclipse中执行以下步骤以导入该项目: eclipse > file > ...
1. 多态性(Polymorphism)的概念,包括静态多态和动态多态。 2. 静态联编(Static Binding)和动态联编(Dynamic Binding)的区别及其在C++中的应用。 3. 函数重载的规则和实现方式,以及如何在基类和派生类中进行...
《popGen32:微卫星遗传多态性统计软件详解》 在生物学研究领域,特别是遗传学和进化生物学中,微卫星(Microsatellites)因其高度多态性而成为重要的遗传标记。微卫星,又称简单重复序列(Simple Sequence Repeats...
论文中提到的NPs(nucleotide polymorphism sites,核苷酸多态性位点)261、467、802、803和1061是与ABO等位基因相关的特定变异位置。 2. **模板筛选**:为了进行后续的嵌合体分析,研究者从76例体检者中筛选出携带...
Java多态性是面向对象编程中的一个重要概念,它允许我们以一种统一的方式来处理不同类型的对象。在Java中,多态性主要体现在两个方面:方法的重载(Overloading)和方法的重写(Overriding)。这里我们将深入探讨这...
Java多态性实践指南 在Java编程语言中,多态性是面向对象设计的一个核心特性,它允许我们使用一个类的引用操作其子类的对象,从而实现代码的灵活性和可扩展性。本实践指南将深入探讨Java中的多态性概念,并通过实际...
Java多态性是面向对象编程中的一个重要特性,它允许我们使用父类引用指向子类对象,从而实现一种灵活的代码设计。在这个“java多态性练习”中,我们将深入探讨多态性的概念、抽象类(abstract class)以及它们在实际...
Java多态性是面向对象编程中的一个重要概念,它在Java中起到了关键的作用,使得程序更加灵活,可扩展性更强。本实验将深入探讨Java多态性的核心原理与应用。 首先,我们要理解多态性的基本含义。多态性...
which has evolved in Volume 2 of the second edition of Thinking in C++, and you’ll also find a chapter on patterns in the first edition of Thinking in Java (I took it out of the second edition ...
反射多态性 该项目为PE图像的自我修改提供了各种实用程序,目的是可以将其合并到外部项目中。 该文档可。 概述 Reflective Polymorphism项目当前由以下两个组件组成,每个组件都包含在各自的.c / .h文件中,并且...
多态性(Polymorphism)是程序设计中的一种重要概念,它允许我们使用一个父类类型的引用来调用子类的方法,从而实现不同对象对同一方法的不同响应。 ### 一、多态的形式和体现 **1. 为什么需要多态性** 在实际...
随着高通量的单核苷酸多态性(Single Nucleotide Polymorphism, SNP)检测技术的发展,世界各地的实验研究积累了大量的SNP数据,但是目前尚无一个全面综合的SNP数据库。SNP在致病基因发现、司法鉴定、个体化医疗等...
Java 泛型与多态性是Java编程中的两个核心概念,它们在构建高效、可维护的代码中扮演着至关重要的角色。在这个名为"Java-Polymorphism-Assignment-2"的项目中,我们将深入探讨这两个主题,理解它们的原理以及如何在...
多态性(Polymorphism)在C++中表现为同一个消息(即成员函数的调用)在不同类型的对象上会产生不同的行为。这种现象源自于不同类型的对象对相同操作的处理方式各异。例如,运算符“+”可以用于整型、浮点型和双精度...
在C++编程语言中,多态性(Polymorphism)是一种核心特性,它允许我们使用一个接口来表示多种类型的数据,或者使用同一个函数调用来处理不同类型的对象。多态性提供了代码的灵活性、可扩展性和重用性,是面向对象...
Designed as a guidebook for those who want to become a Java developer, Java 7: A Comprehensive Tutorial discusses the essential Java programming topics that you need to master in order teach other ...
### 函数式编程中的过载与高阶多态性 #### 概述 本文档《函数式编程中的过载与高阶多态性》由马克·P·琼斯撰写,旨在探讨函数式编程语言中的一些高级特性,特别是针对Hindley/Milner类型系统的扩展。该文档关注的...