Java JVM 多态(动态绑定)
Java JVM 多态(动态绑定)
@author ixenos
摘要:绑定、动态绑定实现多态、多态的缺陷、纯继承与扩展接口、向下转型与RTTI
绑定
将一个方法的调用和一个方法的主体关联起来,称作(方法调用)绑定:
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. 动态绑定(后期绑定或运行时绑定)常被指称为多态
多态的缺陷
“覆盖”私有方法时:
public class Pri{
private void f(){ System.out.println("I'm private f()"); }
public static void main(String[] args){
Pri pri = new Pub(); //向上转型,动态调用Pub类方法
pri.f();
}
}
class Pub{
public void f(){ System.out.println("I'm public f()"); }
}
----------------------------------
输出: 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是看不到的;
若Super.field是public且子类域覆盖了父类域,Sub的默认域就是子类域,要调用父类域要用super.field(field替换为父类域引用变量)
静态方法:静态方法是与类相关联的而不是与对象
final方法:final阻止子类重写,但可重载
纯继承与扩展接口
纯继承:
导出类只覆盖基类的方法,导出类只具有和基类相同的接口,此时导出类可以完全代替基类,基类可以接受发送给导出类的任何信息,我们只需要从导出类向上转型,永远不需要知道正在处理的对象的确切类型,这是通过多态(动态绑定)处理的
Shape{
draw(){}
erase(){}
}
Circle extends Shape{
draw(){}
erase(){}
}
Square extends Shape{
draw(){}
erase(){}
}
Triangle extends Shape{
draw(){}
erase(){}
}
扩展接口:
扩展导出类的功能,但是,导出类中接口的扩展部分不能被基类访问,因此,一旦扩展接口的导出类对象向上转型,就不能调用那些新方法
Useful{
draw(){}
erase(){}
}
MoreUseful extends Useful{
draw(){}
erase(){}
//扩展接口
freak(){}
flyme(){}
}
向下转型与RTTI
向上转型:
向上转型是安全的,因为基类不会具有大于导出类的接口
向下转型:
向下转型是不安全的,由“运行时类型识别”RTTI(Run-Time Type Identification)来确保向下转型正确性
Useful{
draw(){}
erase(){}
}
MoreUseful extends Useful{
draw(){}
erase(){}
//扩展接口
freak(){}
flyme(){}
}
public class RTTI{
public static void mian(String[] args){
Useful[] x = { new Useful() , new MoreUseful() };
// ((MoreUseful)x[0]).freak(); //ERROR 向下转型失败,因为x[0]对象不具有MoreUseful类中的扩展接口
((MoreUseful)x[1]).freak(); //向下转型成功/RTTI
RTTI行为示例
一个基类引用的对象,如果想访问特定导出类对象的扩展接口,就可以尝试向下转型,但是该基类对象本身必须是导出类类型或者其子类型,才能转型成功
Java JVM 多态(动态绑定)的更多相关文章
- [转帖]java基础学习总结——多态(动态绑定)
https://www.cnblogs.com/xdp-gacl/p/3644035.html 多态的概念 java基础学习总结——多态(动态绑定) 一.面向对象最核心的机制——动态绑定,也叫多态
- Java静态绑定与动态绑定
程序绑定的概念: 绑定指的是一个方法的调用与方法所在的类(方法主体)关联起来.对java来说,绑定分为静态绑定和动态绑定:或者叫做前期绑定和后期绑定. 静态绑定: 在程序执行前方法已经被绑定(也就是说 ...
- Java静态绑定和动态绑定
程序绑定的概念: 绑定指的是一个方法的调用与方法所在的类(方法主体)关联起来.对java来说,绑定分为静态绑定和动态绑定:或者叫做前期绑定和后期绑定 静态绑定(早绑定 编译器绑定): 在程序执行前方法 ...
- 关于java中多态的理解
java三大特性:封装,继承,多态. 多态是java的非常重要的一个特性: 那么问题来了:什么是多态呢? 定义:指允许不同类的对象对同一消息做出响应.即同一消息可以根据发送对象的不同而采用多种不同的行 ...
- 个人对Java中多态的一些简单理解
什么是多态 面向对象的三大特性:封装.继承.多态.从一定角度来看,封装和继承几乎都是为多态而准备的.这是我们最后一个概念,也是最重要的知识点. 多态的定义:指允许不同类的对象对同一消息做出响应.即同一 ...
- Java中多态的一些简单理解
什么是多态 .面向对象的三大特性:封装.继承.多态.从一定角度来看,封装和继承几乎都是为多态而准备的.这是我们最后一个概念,也是最重要的知识点. .多态的定义:指允许不同类的对象对同一消息做出响应.即 ...
- Java(15) 多态
一.多态 1.1 多态:一个事物的多种形态 1.2 多态的表现 行为(方法)多态:重写和重载. 引用多态(动态绑定):编译时的类型和运行时不一致的这种现象 例:父类 = 子类 Pet p = ne ...
- 从虚拟机指令执行的角度分析JAVA中多态的实现原理
从虚拟机指令执行的角度分析JAVA中多态的实现原理 前几天突然被一个"家伙"问了几个问题,其中一个是:JAVA中的多态的实现原理是什么? 我一想,这肯定不是从语法的角度来阐释多态吧 ...
- JAVA & C++ 多态
多态,也叫动态绑定. Java: class A { public void f1() { System.out.println("A:f1"); } public void f2 ...
随机推荐
- python爬虫框架scrapy初试(二点一)
功能:爬取某网站部分新闻列表和对应的详细内容. 列表页面http://www.zaobao.com/special/report/politic/fincrisis 实现代码: import scra ...
- java 文件下载支持中文名称
/** * 文件下载 * @param filePath 文件路径 * @param fileName 文件名称 */ public void download(String filePath,Str ...
- <hdu - 1600 - 1601> Leftmost Digit && Rightmost Digit 数学方法求取大位数单位数字
1060 - Leftmost Digit 1601 - Rightmost Digit 1060题意很简单,求n的n次方的值的最高位数,我们首先设一个数为a,则可以建立一个等式为n^n = a * ...
- Java学习日志(20170111)
今日新知识点: 1.关键字volatile sychronized是同步锁,这个之前接触过,在类/方法或代码块前加该修饰词,即可实现线程同步: volatile也是一个修饰符,被volatile修饰的 ...
- JDK各版本新增的主要特性
JDK1.5新特性: 1.自动装箱与拆箱: 2.枚举 3.静态导入,如:import staticjava.lang.System.out 4.可变参数(Varargs) 5.内省(Introspec ...
- Python学习笔记——基础篇【第五周】——正在表达式(re.match与re.search的区别)
目录 1.正在表达式 2.正则表达式常用5种操作 3.正则表达式实例 4.re.match与re.search的区别 5.json 和 pickle 1.正则表达式 语法: import re # ...
- 调皮的R文件,卑鄙的空格
毕业快一年了,由于公司业务需要,开发岗的我做了一年测试.最近,终于要开始转开发了.于是和小伙伴们合作,做一个备忘录apk.由于之前是做java的,而且差不多一年没碰代码了(这一年主要做测试,虽然有写自 ...
- ElasticSearch(8)-分布式搜索
分布式搜索的执行方式 在继续之前,我们将绕道讲一下搜索是如何在分布式环境中执行的. 它比我们之前讲的基础的增删改查(create-read-update-delete ,CRUD)请求要复杂一些. 注 ...
- ElasticSearch(7)-排序
引用自ElaticSearch权威指南 一.排序 相关性排序 默认情况下,结果集会按照相关性进行排序 -- 相关性越高,排名越靠前. 这一章我们会讲述相关性是什么以及它是如何计算的. 在此之前,我们先 ...
- django模板 实现奇偶分行
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...