`
yanchengxiaotubao
  • 浏览: 41680 次
  • 性别: Icon_minigender_1
  • 来自: 苏州
社区版块
存档分类
最新评论

Think in Java - Chatpter 7 polymorphism 多态性

 
阅读更多

多态性:

  • 是什么
  • 怎么做

优点:

  • 代码组织以及可读性均能获得改善
  • 创建“易于扩展”的程序

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 
// 如果内部类是匿名的,那么编译器会简单地生成数字,把它们作为内部类标识符使用

 内部类:

  1. 在单独一个类里表达一个控制框架应用的全部实施细节,从而完整地封装与那个实施有关的所有东西。内部类用与表达多种不同类型的的action,用于解决实际的问题。
  2. 内部类使我们具体的实施变得更加巧妙,因为能方便地访问外部类的任何成员。
  • 大小: 88.8 KB
  • 大小: 56.6 KB
分享到:
评论

相关推荐

    java-program-design-principles-polymorphism

    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 ...

    inheritance-polymorphism-java:演示和理解多态性

    #Java中的多态性和继承###是什么? 一个Java项目,以演示和理解Java中的多态性和继承。 该项目使用Java 7。 ###如何使用? 这是一个eclipse项目,因此要从eclipse中执行以下步骤以导入该项目: eclipse > file > ...

    C++程序设计基础PPT课件8-虚函数与多态性

    1. 多态性(Polymorphism)的概念,包括静态多态和动态多态。 2. 静态联编(Static Binding)和动态联编(Dynamic Binding)的区别及其在C++中的应用。 3. 函数重载的规则和实现方式,以及如何在基类和派生类中进行...

    popGen32-微卫星遗传多态性统计

    《popGen32:微卫星遗传多态性统计软件详解》 在生物学研究领域,特别是遗传学和进化生物学中,微卫星(Microsatellites)因其高度多态性而成为重要的遗传标记。微卫星,又称简单重复序列(Simple Sequence Repeats...

    大数据-算法-ABO基因相关多态性位点在ABO省略CT移植后嵌合体定量分析中的研究.pdf

    论文中提到的NPs(nucleotide polymorphism sites,核苷酸多态性位点)261、467、802、803和1061是与ABO等位基因相关的特定变异位置。 2. **模板筛选**:为了进行后续的嵌合体分析,研究者从76例体检者中筛选出携带...

    java-polymorphism.zip_class A

    Java多态性是面向对象编程中的一个重要概念,它允许我们以一种统一的方式来处理不同类型的对象。在Java中,多态性主要体现在两个方面:方法的重载(Overloading)和方法的重写(Overriding)。这里我们将深入探讨这...

    practice-java-polymorhism.rar_practice

    Java多态性实践指南 在Java编程语言中,多态性是面向对象设计的一个核心特性,它允许我们使用一个类的引用操作其子类的对象,从而实现代码的灵活性和可扩展性。本实践指南将深入探讨Java中的多态性概念,并通过实际...

    java多态性练习

    Java多态性是面向对象编程中的一个重要特性,它允许我们使用父类引用指向子类对象,从而实现一种灵活的代码设计。在这个“java多态性练习”中,我们将深入探讨多态性的概念、抽象类(abstract class)以及它们在实际...

    实验二:java多态性.rar

    Java多态性是面向对象编程中的一个重要概念,它在Java中起到了关键的作用,使得程序更加灵活,可扩展性更强。本实验将深入探讨Java多态性的核心原理与应用。 首先,我们要理解多态性的基本含义。多态性...

    think in java pattern

    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 ...

    reflective-polymorphism:反射多态性

    反射多态性 该项目为PE图像的自我修改提供了各种实用程序,目的是可以将其合并到外部项目中。 该文档可。 概述 Reflective Polymorphism项目当前由以下两个组件组成,每个组件都包含在各自的.c / .h文件中,并且...

    Java面向对象(进阶)- 面向对象特征之三:多态性

    多态性(Polymorphism)是程序设计中的一种重要概念,它允许我们使用一个父类类型的引用来调用子类的方法,从而实现不同对象对同一方法的不同响应。 ### 一、多态的形式和体现 **1. 为什么需要多态性** 在实际...

    论文研究-大规模单核苷酸多态性位点验证的集群解决方案.pdf

    随着高通量的单核苷酸多态性(Single Nucleotide Polymorphism, SNP)检测技术的发展,世界各地的实验研究积累了大量的SNP数据,但是目前尚无一个全面综合的SNP数据库。SNP在致病基因发现、司法鉴定、个体化医疗等...

    Java-Polymorphism-Assignment-2

    Java 泛型与多态性是Java编程中的两个核心概念,它们在构建高效、可维护的代码中扮演着至关重要的角色。在这个名为"Java-Polymorphism-Assignment-2"的项目中,我们将深入探讨这两个主题,理解它们的原理以及如何在...

    C++中的多态性的分析与作用

    多态性(Polymorphism)在C++中表现为同一个消息(即成员函数的调用)在不同类型的对象上会产生不同的行为。这种现象源自于不同类型的对象对相同操作的处理方式各异。例如,运算符“+”可以用于整型、浮点型和双精度...

    C++课程-5_多态性

    在C++编程语言中,多态性(Polymorphism)是一种核心特性,它允许我们使用一个接口来表示多种类型的数据,或者使用同一个函数调用来处理不同类型的对象。多态性提供了代码的灵活性、可扩展性和重用性,是面向对象...

    Java.7.A.Comprehensive.Tutorial

    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 ...

    Functional Programming with Overloading and Higher-Order Polymorphism PDF 英文

    ### 函数式编程中的过载与高阶多态性 #### 概述 本文档《函数式编程中的过载与高阶多态性》由马克·P·琼斯撰写,旨在探讨函数式编程语言中的一些高级特性,特别是针对Hindley/Milner类型系统的扩展。该文档关注的...

Global site tag (gtag.js) - Google Analytics