8.2.1 方法调用绑定
解决的办法就是后期绑定,它的含义就是在运行时根据对象的类型进行绑定。后期绑定也
叫做动态绑定或运行时绑定.如果一种语言想实现后期绑定,就必须具有某种机制,以便在运
行时能判断对象的类型,从而调用恰当的方法。也就是说,编译器一直不知道对象的类型,但
是方法调用机制能找到正确的方法体,并加以调用。后期绑定机制随编程语言的不同而有所不
同,但是只要想一下就会得知,不管怎样都必须在对象中安置某种"类型信息"。
Java中除了static方法和final方法( private方法属于final方法)之外,其他所有的方法都是
后期绑定。这意味着通常情况下,我们不必判定是否应该进行后期绑定一一它会自动发生。
为什么要将某个方法声明为final呢?正如前一章提到的那样,它可以防止其他人覆盖该方
法。但更重要的一点或许是:这样做可以有效地"关闭"动态绑定,或者说,告诉编译器不需
要对其进行动态绑定。这样,编译器就可以为final方法调用生成更有效的代码。然而,大多数
情况下,这样做对程序的整体性能不会有什么改观。所以,最好根据设计来决定是否使用final ,
而不是出于试图提高性能的目的来使用final .
 
8.2.4缺陷:"覆盖"私有方法
例子:
package com.cy.polymorphism;

// Trying to override a private method.
import static com.java.util.Print.*; public class PrivateOverride {
private void f() { print("private f()"); } public static void main(String[] args) {
PrivateOverride po = new Derived();
po.f();
}
} class Derived extends PrivateOverride { public void f() { print("public f()"); }
} /* Output:
private f()
*///:~
我们所期望的输出是public f(). 但是由于private方法被自动认为是final方法, 而且对导出
类是屏蔽的。因此,在这种情况下, Derived类中的f()方法就是一个全新的方怯,既然基类中的
f()方法在子类Derived中不可见,因此甚至也不能被重载。
结论就是:只有非private方法才可以被覆盖;但是还需要密切注意覆盖private方法的现象,
这时虽然编译器不会报错,但是也不会按照我们所期望的来执行。确切地说,在导出类中,对
于基类中的private方法,最好采用不同的名字。
 
8.2.5 缺陷:域与静态方法
一旦你了解了多态机制,可能就会开始认为所有事物都可以多态地发生。然而,只有普通
的方法调用可以是多态的。例如,如果你直接访问某个域,这个访问就将在编译期进行解析,
就像下面的示例所演示的:
package com.cy.polymorphism;

// Direct field access is determined at compile time.
class Super {
public int field = 0;
public int getField() { return field; }
} class Sub extends Super {
public int field = 1;
public int getField() { return field; }
public int getSuperField() { return super.field; }
} public class FieldAccess {
public static void main(String[] args) {
Super sup = new Sub(); // Upcast
System.out.println("sup.field = " + sup.field +", sup.getField() = " + sup.getField()); Sub sub = new Sub();
System.out.println("sub.field = " + sub.field + ", sub.getField() = " +
sub.getField() +", sub.getSuperField() = " + sub.getSuperField());
}
}
/* Output:
sup.field = 0, sup.getField() = 1
sub.field = 1, sub.getField() = 1, sub.getSuperField() = 0
*///:~
当Sub对象转型为Super引用肘,任何域访问操作都将由编译器解析,因此不是多态的。在
本例中,为Super.field和Sub.fleld分配了不同的存储空间。这样, Sub实际上包含两个称为fleld
的域:它自己的和它从Super处得到的.然而,在引用Sub中的field时所产生的默认域并非Super
版本的field域。因此,为了得到Super.field ,必须显式地指明super.field .
尽管这看起来好像会成为一个容易令人棍淆的问题,但是在实践中,它实际上从来不会发
生.首先,你通常会将所有的域都设置成private,因此不能直接访问它们,其副作用是只能调
用方法来访问.另外,你可能不会对基类中的域和导出类中的域赋予相同的名字,因为这种做
法容易令人混淆。
 
如果某个方法是静态的,它的行为就不具有多态性:
package com.cy.polymorphism;

// Static methods are not polymorphic.
class StaticSuper {
public static String staticGet() {
return "Base staticGet()";
}
public String dynamicGet() {
return "Base dynamicGet()";
}
} class StaticSub extends StaticSuper {
public static String staticGet() {
return "Derived staticGet()";
}
public String dynamicGet() {
return "Derived dynamicGet()";
}
} public class StaticPolymorphism {
public static void main(String[] args) {
StaticSuper sup = new StaticSub(); // Upcast
System.out.println(sup.staticGet());
System.out.println(sup.dynamicGet());
}
}
/* Output:
Base staticGet()
Derived dynamicGet()
*///:~
静态方法是与类,而并非与单个的对象相关联的。
 

8.3 构造器和多态
通常,构造器不同于其他种类的方法.涉及到多态时仍是如此.尽管构造器并不具有多态
性(它们实际上是static方法,只不过该static声明是隐式的) .但还是非常有必要理解构造器怎
样通过多态在复杂的层次结构中运作,这一理解将有助于大家避免一些令人不快的困扰.
8.3.1 构造器的调用顺序
让我们来看下面这个例子,它展示组合、继承以及多态在构建顺序上的作用:
package com.cy.polymorphism;

// Order of constructor calls.
import static com.java.util.Print.*; class Meal {
Meal() { print("Meal()"); }
} class Bread {
Bread() { print("Bread()"); }
} class Cheese {
Cheese() { print("Cheese()"); }
} class Lettuce {
Lettuce() { print("Lettuce()"); }
} class Lunch extends Meal {
Lunch() { print("Lunch()"); }
} class PortableLunch extends Lunch {
PortableLunch() { print("PortableLunch()");}
} public class Sandwich extends PortableLunch {
private Bread b = new Bread();
private Cheese c = new Cheese();
private Lettuce l = new Lettuce();
public Sandwich() { print("Sandwich()"); } public static void main(String[] args) {
new Sandwich();
}
}
/* Output:
Meal()
Lunch()
PortableLunch()
Bread()
Cheese()
Lettuce()
Sandwich()
*///:~
在这个例子中,用其他类创建了一个复杂的类,而且每个类都有一个声明它自己的构造器.
其中最重要的类是Sandwich ,它反映了三层继承(若将自Object的隐含继承也算在内,就是四,
层)以及三个成员对象。当在main()里创建一个Sandwich对象后,就可以看到输出结果。这也
表明了这一复杂对象调用构造器要遵照下面的顺序:
1) 调用基类构造器。这个步骤会不断地反复递归下去,首先是构造这种层次结构的根,然
后是下一层导出类,等等,直到最低层的导出类。
2) 按声明顺序调用成员的初始化方法。
3) 调用导出类构造器的主体。
构造器的调用顺序是很重要的。当进行继承肘,我们已经知道基类的一切,并且可以访问
基类中任何声明为public和protected的成员。这意味着在导出类中,必须假定基类的所有成员
都是有效的。一种标准方法是,构造动作一经发生,那么对象所有部分的全体成员都会得到构
建。然而,在构造器内部,我们必须确保所要使用的成员都已经构建完毕。为确保这一目的,
唯一的办法就是首先调用基类构造器。那么在进入导出类构造器时,在基类中可供我们访问的
成员都已得到初始化。此外,知道构造器中的所有成员都有效也是因为, 当成员对象在类内进
行定义的时候(比如上例中的b 、c和l ) ,只要有可能,就应该对它们进行初始化(也就是说,
通过组合方法将对象置于类内)。若遵循这一规则,那么就能保证所有基类成员以及当前对象的
成员对象都被初始化了。但遗憾的是,这种做法并不适用于所有情况,这一点我们会在下一节
中看到。
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
----------------

ThinkJava-多态的更多相关文章

  1. Java中的多态

    1.多态:具有表现多种形态的能力的特征 父类: public abstract class Animal {public abstract void Say();} 子类: public class ...

  2. C# 工厂模式+虚方法(接口、抽象方法)实现多态

    面向对象语言的三大特征之一就是多态,听起来多态比较抽象,简而言之就是同一行为针对不同对象得到不同的结果,同一对象,在不同的环境下得到不同的状态. 实例说明: 业务需求:实现一个打开文件的控制台程序的d ...

  3. C#非常重要基础之多态

    前几天看了一位同志的博客,写的是关于他自己去支付宝面试的经历.过程大体是这样的:问答的时候,前面部分,作者都应答如流,说起自己经验如何之丰富,最后面试官问了作者一个问题:请简述多态的概念和作用.结果这 ...

  4. C++多态详解

    多态是面向对象的程序设计的关键技术.多态:调用同一个函数名,可以根据需要但实现不同的功能.多态体现在两个方面,我们以前学过的编译时的多态性(函数重载)和现在我们这一章将要学习的运行时的多态性(虚函数) ...

  5. 【那些年关于java多态应用】

    1.多态:具有表现多种形态的能力的特征 父类: public abstract class Animal { public abstract void Say();} 子类: public class ...

  6. JAVA多态

    多态是指当系统A访问系统B的服务时,系统B可以通过多种方式来提供服务,而这一切对系统A是透明的.比如动物园的饲养员能够给各种各样的动物喂食.下图显示了饲养员Feeder,食物Food和动物Animal ...

  7. C#多态“说来也说”——逻辑层BLL中的多态使用

    本文版权归博客园和作者吴双本人共同所有.欢迎转载,转载和爬虫请注明原文地址 http://www.cnblogs.com/tdws/p/5861842.html 昨天晚上,有个朋友说学了好久,依然没搞 ...

  8. java多态的理解

    面向对象语言中的类有三个特征,封装.继承.多态.封装与继承很好理解,那什么是多态呢? 1.什么是多态? 多态的定义:指允许不同类的对象对同一消息做出响应.即同一消息可以根据发送对象的不同而采用多种不同 ...

  9. java中如何实现多态

    复习基础知识 多态,就是重载和重写.重载发生在一个类中.重写发生在子类,意思就是子类重写父类相同名称的方法.刚学语言有的东西,不必搞得那么清楚,只有知道怎么用就行了,有的问题你要想真正把它搞得很懂,短 ...

  10. OC多态

    要点: 1.多种形态,引用的多种形态对于一个引用变量,可以指向任何类的对象.对于一个父类的引用(类与类之间有一种继承关系),可以指向子类,也可以指向本类,指向的类型不同.当通过此引用向对象发送消息,调 ...

随机推荐

  1. 《Python》 property、classmethod、staticmethod、isinstance、issubclass

    一.property property是一个装饰器函数 装饰器函数的使用方法:在函数.方法.类的上面一行直接@装饰器的名字 装饰器的分类: 1.装饰函数 2.装饰方法:property 3.装饰类 i ...

  2. 玩转X-CTR100 l STM32F4 l 电机正交编码器

    我造轮子,你造车,创客一起造起来!塔克创新资讯[塔克社区 www.xtark.cn ][塔克博客 www.cnblogs.com/xtark/ ] 本文介绍X-CTR100控制器的电机正交编码器,X- ...

  3. 量身打造自己的MyEclipse(多图)

    迎新年 贺元旦MyEclipse推新版 在线订购低至 7.5 折!截止1月31号(活动期间在线下单的客户才可享受此优惠,过期恢复原价) 立即抢购 1.量身打造你自己的MyEclipse MyEclip ...

  4. Centos中查看系统信息的常用命令

    系统日志文件(可以通过cat或tail命令来查看) /var/log/message 系统启动后的信息和错误日志,是Red Hat Linux中最常用的日志之一 /var/log/secure 与安全 ...

  5. 各大公司java后端开发面试题

    各大公司Java后端开发面试题总结 ThreadLocal(线程变量副本)Synchronized实现内存共享,ThreadLocal为每个线程维护一个本地变量.采用空间换时间,它用于线程间的数据隔离 ...

  6. Use JAWS 14 in a VM

    We were not able to run the JAWS 14 app in a Virtual Machine after the installation is completed, th ...

  7. git的使用基础

    /*游戏或者运动才能让我短暂的忘记心痛,现如今感觉学习比游戏和运动还重要——曾少锋*/ 在Git-Bash中配置自己的名字和Email: git config --global user.name & ...

  8. js 如何控制文本域输入内容在一定间隔时间段才触发事件查询相关数据

    <script>var flag = 0;var t;function openFlag () { t = setTimeout(function(){flag = 1; dosometh ...

  9. qwb的骚扰

    题目描述 自从学姐拒绝了qwb之后,qwb开始了疯狂的骚扰.qwb来到了一个公共电话亭,他摸摸口袋只有n元钱. 已知该公用电话的规则是,前3分钟一共收费x元(不到3分钟也要收x元),超过3分钟每分钟收 ...

  10. hive split 注意事项

    hive字符串分割函数 split(str, regex) - Splits str around occurances that match regexTime taken: 0.769 secon ...