Java 多态(动态绑定)

Java 多态(动态绑定)

@author ixenos

绑定



将一个方法的调用和一个方法的主体关联起来,称作(方法调用)绑定:

  1.前期绑定:在程序执行前绑定(由编译器和连接程序实现);

  2.后期绑定:在运行时根据对象的类型绑定(也称动态绑定运行时绑定);

a) 实现条件:能在运行时判断对象的类型,从而关联对应主体,调用其方法

b) 编译器一直不知道真实对象类型,只将其认作引用变量的类型且知道有继承关系

c) Java中除了static方法final方法private方法属于final方法)是前期绑定之外,其他所有的方法都是后期绑定

动态绑定实现多态:



多态作用:

  1.消除类型之间的耦合关系,使我们可以编写只与基类打交道的程序代码了(比如List<String> = new ArrayList<>();)

  2.使程序有可扩展性,我们可从通用的基类继承出新的数据类型从而添加新的功能,而不需要改变对应接受参数的方法,只与基类接口通信

动态绑定:

  1. 假设B extends A,若使A = new B()向上转型,则编译器认为这是合理的协变,编译通过!但此时编译器只知道B是A的子类,无法得知B的具体类型!

  2. 运行时,Java的后期绑定机制判定该对象new B()的运行时类型为B,所以方法的调用策略是从B类中调用相应方法(static、final方法除外)

  3. 动态绑定(后期绑定或运行时绑定)常被指称为多态

多态的缺陷



 “覆盖”私有方法时

 1 public class Pri{
 2     private void f(){ System.out.println("I‘m private f()"); }
 3
 4     public static void main(String[] args){
 5         Pri pri = new Pub(); //向上转型,动态调用Pub类方法
 6         pri.f();
 7     }
 8 }
 9
10 class Pub{
11     public void f(){ System.out.println("I‘m public f()"); }
12 }
13
14
15 ----------------------------------
16 输出:  I‘m private f()

  我们期望输出是public f(),但private方法被认为是final方法,这对导出类(子类)是屏蔽的,因此没有重写和重载!

  此时,Pub类中的f()方法就是一个全新的方法,两个方法存在不同类中

  结论:虽然只有非private方法可以覆盖,但也要注意这种试图覆盖private方法的行为,毕竟编译器不会报错(因为根本就是创建一个新的方法)导致方法不按照初衷来执行

不适用多态的范围:域、静态方法与final方法

  只有普通方法的调用可以是多态的

  域:当Sub对象转型为Super引用时,任何域访问操作都由编译器解析,因此不是多态的;而此时将为Super.field和Sub.field分配不同存储空间,此时Sub包含两个域,他自己的和从super得到的,若Super.field是private的,那么Sub是看不到的,若public且子类域覆盖了父类域,Sub的默认域就是子类域,要调用父类域要用super.field(field替换为父类域引用变量)

  静态方法:静态方法是与类相关联的而不是与对象

纯继承与扩展接口



纯继承:导出类只覆盖基类的方法,导出类只具有和基类相同的接口,此时导出类可以完全代替基类,基类可以接受发送给导出类的任何信息,我们只需要从导出类向上转型,永远不需要知道正在处理的对象的确切类型,这是通过多态(动态绑定)处理的

 1 Shape{
 2     draw(){}
 3     erase(){}
 4 }
 5
 6 Circle extends Shape{
 7     draw(){}
 8     erase(){}
 9 }
10
11 Square extends Shape{
12     draw(){}
13     erase(){}
14 }
15
16 Triangle extends Shape{
17     draw(){}
18     erase(){}
19 }

扩展接口:扩展导出类的功能,但是,导出类中接口的扩展部分不能被基类访问,因此,一旦导出类对象向上转型,就不能调用那些新方法

 1 Useful{
 2     draw(){}
 3     erase(){}
 4 }
 5
 6 MoreUseful extends Useful{
 7     draw(){}
 8     erase(){}
 9
10     //扩展接口
11     freak(){}
12     flyme(){}
13 }

向下转型与RTTI



向上转型:向上转型是安全的,因为基类不会具有大于导出类的接口

向下转型:向下转型是不安全的,由“运行时类型识别”RTTI(Run-Time Type Identification)来确保向下转型正确性

 1 Useful{
 2      draw(){}
 3      erase(){}
 4  }
 5
 6 MoreUseful extends Useful{
 7      draw(){}
 8      erase(){}
 9
10      //扩展接口
11      freak(){}
12      flyme(){}
13  }
14
15 public class RTTI{
16     public static void mian(String[] args){
17         Useful[] x = { new Useful() , new MoreUseful() };
18
19        // ((MoreUseful)x[0]).freak(); //ERROR 向下转型失败,因为x[0]对象不具有MoreUseful类中的扩展接口
20
21         ((MoreUseful)x[1]).freak();  //向下转型成功/RTTI
22
23
24
25
26
27
28
29   

RTTI行为示例

一个基类引用的对象,如果想访问特定导出类对象的扩展接口,就可以尝试向下转型,但是该基类对象本身必须是导出类类型或者其子类型,才能转型成功

  

时间: 2024-08-02 02:43:56

Java 多态(动态绑定)的相关文章

Java多态对象的类型转换和动态绑定

Java多态对象的类型转换这里所说的对象类型转换,是指存在继承关系的对象,不是任意类型的对象.当对不存在继承关系的对象进行强制类型转换时,java 运行时将抛出 java.lang.ClassCastException 异常. 在继承链中,我们将子类向父类转换称为"向上转型",将父类向子类转换称为"向下转型". 很多时候,我们会将变量定义为父类的类型,却引用子类的对象,这个过程就是向上转型.程序运行时通过动态绑定来实现对子类方法的调用,也就是多态性. 然而有些时候为

Java多态之动态绑定

目录 Java多态之动态绑定 引用变量的类型 编译时类型 运行时类型 方法绑定 静态绑定 动态绑定 方法表 Java多态之动态绑定 上篇回顾:多态是面向对象程序设计非常重要的特性,它让程序拥有 更好的可读性和可扩展性. 发生在继承关系中. 需要子类重写父类的方法. 父类类型的引用指向子类类型的对象. 自始至终,多态都是对于方法而言,对于类中的成员变量,没有多态的说法. 上篇说到:一个基类的引用变量接收不同子类的对象将会调用子类对应的方法,这其实就是动态绑定的过程.在理解动态绑定之前,先补充一些概

Java多态小总结

多态,又可以称为动态绑定,即在运行时确定类型,比如: 1 class A { 2 void draw(){ 3 //输出“A” 4 } 5 } 6 class B { 7 void draw(){ 8 //输出“B” 9 } 10 11 } 这种关系里,如果调用A a = new B(); 此时,被称为向上转型,a的类型可能在很早之前被生命,而在这时候被明确指明是其子类型, 我们如果要去调用draw()方法的时候,会调用输出“B”,这样,便是Java中的“多态”.我们称其为“向上转型”. 但是,

java多态/重载方法——一个疑难代码引发的讨论

直接上代码,看这个代码发现自己的基础有多差了.参考 http://www.cnblogs.com/lyp3314/archive/2013/01/26/2877205.html和http://hxraid.iteye.com/blog/428891 以及 <深入Java虚拟机> 这个问题是java获取动态绑定最终的方法?涉及的知识有继承.多态.重载.方法调用的整合. public class Poly { public static void main(String[] args) { A a

Java 多态——与C++的比较

学习了Java和C++之后,由于长期不使用C++,而java的基础知识掌握不牢,现在已经搞不清java多态了.现在先来谈谈java多态,稍后有时间再更新C++的多态,并进行比较~ 一. Java的多态 首先什么是Java的多态? 多态是同一个行为具有多个不同表现形式或形态的能力.多态就是同一个接口,使用不同的实例而执行不同操作. 实现多态的技术称为:动态绑定(dynamic binding),是指在执行期间判断所引用对象的实际类型,根据其实际的类型调用其相应的方法. 多态的作用:消除类型之间的耦

对java多态的理解

java多态,如何理解父类引用指向子类对象 要理解多态性,首先要知道什么是"向上转型". 我定义了一个子类Cat,它继承了Animal类,那么后者就是前者是父类.我可以通过 Cat c = new Cat(); 实例化一个Cat的对象,这个不难理解. 但当我这样定义时: Animal a = new Cat(); 表示定义了一个Animal类型的引用,指向新建的Cat类型的对象.由于Cat是继承自它的父类Animal,所以Animal类型的引用是可以指向Cat类型的对象的. 那么这样做

Java多态的底层原理

作为一门面向对象语言,Java 拥有封装.继承.多态三大特性.多态就是允许不同类的对象对同一消息做出响应.基于多态,可以消除一些类型耦合关系,实现可替换.可扩充.Java 中使用多态特性的方法主要有,实现一个接口,实现抽象类的一个方法,覆盖父类的一个方法. Java多态的底层原理 多态的底层实现是动态绑定,即在运行时才把方法调用与方法实现关联起来.动态绑定涉及很多 JVM 的细节,全部写清楚需要很大篇幅,因此本文仅简单描述,后续会有其他文章就其中的细节一一分享. 静态绑定与动态绑定 JVM 的方

Java多态之向下转型

目录 Java多态之向下转型 强制类型转换 instanceof Java多态之向下转型 往期回顾:我们学习了向上转型和动态绑定的概念,可以知道在继承关系中,将一个子类对象赋值给父类的引用变量,调用父类的方法,在实际运行时,就可以根据子类中重写的方法执行不同的操作.其中有一个弊端,就是在向上转型的过程中,其实丢失了一部分子类特有的功能,毕竟它只允许调用父类中的方法.那么,如何在这时调用子类中的方法呢,这时就需要与向上转型相对应的方法,就是所谓的:向下转型. 向上转型是自动就能完成的,向下转型则需

Java的动态绑定机制

Java的动态绑定又称为运行时绑定.意思就是说,程序会在运行的时候自动选择调用哪儿个方法. 一.动态绑定的过程: 例子: public class Son extends Father Son son = new Son(); son.method(); 1. 首先,编译器根据对象的声明类型和方法名,搜索相应类(Son)及其父类(Father)的"方法表",找出所有访问属性为public的method方法. 可能存在多个方法名为method的方法,只是参数类型或数量不同. 2. 然后,