第七章:多态
主要内容之一:静态绑定和动态绑定
对于Shape s=new Circle(),Shape是父类,Circle继承Shape。自然是一个正确的表达式,我对于这个表达式的理解是:new Circle()创建了Circle的对象存储在堆中,并返回了Circle的引用,当该引用赋值给s后,也就是向上转型,Circle类有可能将缩小方法范围,但肯定包含了Shape的方法,那么实际上s指向的是一个减少了方法的Circle对象(为了适应Shape),所以用s调用某个方法时,实际上还是根据new Circle()返回的引用去找到对象,在通过该对象找到Circle类的方法表,调用Circle类方法表中的对应的方法区里的某个方法,而不是Shape类方法区中的某个方法。
另外要注意的一点是,每个类都有自己的方法区,Shape类里的方法f()方法和Circle类里的f()方法,虽然后者是复写前者得来,但是在方法区里实现的方式可能完全不同。在创建对象时,JVM会将Shape类中方法表的索引记录到Circle类的方法表中,但这只是一个”指针”的传递,并非方法的传递。
以下是网上详细的解释,博客地址:http://hxraid.iteye.com/blog/428891
在Java方法调用的过程中,JVM是如何知道调用的是哪个类的方法源代码? 这里面到底有什么内幕呢? 这篇文章我们就将揭露JVM方法调用的静态(static binding) 和动态绑定机制(auto binding) 。
★ 静态绑定机制
Java代码
1. //被调用的类
2. package hr.test;
3. class Father{
4. public static void f1(){
5. System.out.println("Father— f1()");
6. }
7. }
8. //调用静态方法
9. import hr.test.Father;
10. public class StaticCall{
11. public static void main(){
12. Father.f1(); //调用静态方法
13. }
14. }
上面的源代码中执行方法调用的语句(Father.f1())被编译器编译成了一条指令:invokestatic #13。我们看看JVM是如何处理这条指令的
(1) 指令中的#13指的是StaticCall类的常量池中第13个常量表的索引项(关于常量池详见《Class文件内容及常量池 》)。这个常量表(CONSTATN_Methodref_info ) 记录的是方法f1信息的符号引用(包括f1所在的类名,方法名和返回类型)。JVM会首先根据这个符号引用找到方法f1所在的类的全限定名: hr.test.Father。
(2) 紧接着JVM会加载、链接和初始化Father类。
(3) 然后在Father类所在的方法区中找到f1()方法的直接地址,并将这个直接地址记录到StaticCall类的常量池索引为13的常量表中。这个过程叫常量池解析 ,以后再次调用Father.f1()时,将直接找到f1方法的字节码。
(4) 完成了StaticCall类常量池索引项13的常量表的解析之后,JVM就可以调用f1()方法,并开始解释执行f1()方法中的指令了。
通过上面的过程,我们发现经过常量池解析之后,JVM就能够确定要调用的f1()方法具体在内存的什么位置上了。实际上,这个信息在编译阶段就已经在StaticCall类的常量池中记录了下来。这种在编译阶段就能够确定调用哪个方法的方式,我们叫做 静态绑定机制 。
除了被static 修饰的静态方法,所有被private 修饰的私有方法、被final 修饰的禁止子类覆盖的方法都会被编译成invokestatic指令。另外所有类的初始化方法<init>和<clinit>会被编译成invokespecial指令。JVM会采用静态绑定机制来顺利的调用这些方法。
★ 动态绑定机制
Java代码
1. package hr.test;
2. //被调用的父类
3. class Father{
4. public void f1(){
5. System.out.println("father-f1()");
6. }
7. public void f1(int i){
8. System.out.println("father-f1() para-int "+i);
9. }
10. }
11. //被调用的子类
12. class Son extends Father{
13. public void f1(){ //覆盖父类的方法
14. System.out.println("Son-f1()");
15. }
16. public void f1(char c){
17. System.out.println("Son-s1() para-char "+c);
18. }
19. }
20.
21. //调用方法
22. import hr.test.*;
23. public class AutoCall{
24. public static void main(String[] args){
25. Father father=new Son(); //多态
26. father.f1(); //打印结果: Son-f1()
27. }
28. }
上面的源代码中有三个重要的概念:多态(polymorphism) 、方法覆盖 、方法重载 。打印的结果大家也都比较清楚,但是JVM是如何知道f.f1()调用的是子类Sun中方法而不是Father中的方法呢?在解释这个问题之前,我们首先简单的讲下JVM管理的一个非常重要的数据结构——方法表 。
在JVM加载类的同时,会在方法区中为这个类存放很多信息(详见《Java 虚拟机体系结构 》)。其中就有一个数据结构叫方法表。它以数组的形式记录了当前类及其所有超类的可见方法字节码在内存中的直接地址 。下图是上面源代码中Father和Sun类在方法区中的方法表:
上图中的方法表有两个特点:(1) 子类方法表中继承了父类的方法,比如Father extends Object。 (2) 相同的方法(相同的方法签名:方法名和参数列表)在所有类的方法表中的索引相同。比如Father方法表中的f1()和Son方法表中的f1()都位于各自方法表的第11项中。
对于上面的源代码,编译器首先会把main方法编译成下面的字节码指令:
多态调用的字节码指令代码
1. 0 new hr.test.Son [13] //在堆中开辟一个Son对象的内存空间,并将对象引用压入操作数栈
2. 3 dup
3. 4 invokespecial #7 [15] // 调用初始化方法来初始化堆中的Son对象
4. 7 astore_1 //弹出操作数栈的Son对象引用压入局部变量1中
5. 8 aload_1 //取出局部变量1中的对象引用压入操作数栈
6. 9 invokevirtual #15 //调用f1()方法
7. 12 return
其中invokevirtual指令的详细调用过程是这样的:
(1) invokevirtual指令中的#15指的是AutoCall类的常量池中第15个常量表的索引项。这个常量表(CONSTATN_Methodref_info ) 记录的是方法f1信息的符号引用(包括f1所在的类名,方法名和返回类型)。JVM会首先根据这个符号引用找到调用方法f1的类的全限定名: hr.test.Father。这是因为调用方法f1的类的对象father声明为Father类型。
(2) 在Father类型的方法表中查找方法f1,如果找到,则将方法f1在方法表中的索引项11(如上图)记录到AutoCall类的常量池中第15个常量表中(常量池解析 )。这里有一点要注意:如果Father类型方法表中没有方法f1,那么即使Son类型中方法表有,编译的时候也通过不了。因为调用方法f1的类的对象father的声明为Father类型。
(3) 在调用invokevirtual指令前有一个aload_1指令,它会将开始创建在堆中的Son对象的引用压入操作数栈。然后invokevirtual指令会根据这个Son对象的引用首先找到堆中的Son对象,然后进一步找到Son对象所属类型的方法表。过程如下图所示:
(4) 这是通过第(2)步中解析完成的#15常量表中的方法表的索引项11,可以定位到Son类型方法表中的方法f1(),然后通过直接地址找到该方法字节码所在的内存空间。
很明显,根据对象(father)的声明类型(Father)还不能够确定调用方法f1的位置,必须根据father在堆中实际创建的对象类型Son来确定f1方法所在的位置。这种在程序运行过程中,通过动态创建的对象的方法表来定位方法的方式,我们叫做 动态绑定机制 。
上面的过程很清楚的反映出在方法覆盖的多态调用的情况下,JVM是如何定位到准确的方法的。但是下面的调用方法JVM是如何定位的呢?(仍然使用上面代码中的Father和Son类型)
Java代码
1. public class AutoCall{
2. public static void main(String[] args){
3. Father father=new Son();
4. char c='a';
5. father.f1(c); //打印结果:father-f1() para-int 97
6. }
7. }
问题是Fahter类型中并没有方法签名为f1(char)的方法呀。但打印结果显示JVM调用了Father类型中的f1(int)方法,并没有调用到Son类型中的f1(char)方法。
根据上面详细阐述的调用过程,首先可以明确的是:JVM首先是根据对象father声明的类型Father来解析常量池的(也就是用Father方法表中的索引项来代替常量池中的符号引用)。如果Father中没有匹配到"合适" 的方法,就无法进行常量池解析,这在编译阶段就通过不了。
那么什么叫"合适"的方法呢?当然,方法签名完全一样的方法自然是合适的。但是如果方法中的参数类型在声明的类型中并不能找到呢?比如上面的代码中调用father.f1(char),Father类型并没有f1(char)的方法签名。实际上,JVM会找到一种“凑合”的办法,就是通过 参数的自动转型 来找 到“合适”的 方法。比如char可以通过自动转型成int,那么Father类中就可以匹配到这个方法了 (关于Java的自动转型问题可以参见《【解惑】Java类型间的转型》)。但是还有一个问题,如果通过自动转型发现可以“凑合”出两个方法的话怎么办?比如下面的代码:
Java代码
1. class Father{
2. public void f1(Object o){
3. System.out.println("Object");
4. }
5. public void f1(double[] d){
6. System.out.println("double[]");
7. }
8.
9. }
10. public class Demo{
11. public static void main(String[] args) {
12. new Father().f1(null); //打印结果: double[]
13. }
14. }
null可以引用于任何的引用类型,那么JVM如何确定“合适”的方法呢。一个很重要的标准就是:如果一个方法可以接受传递给另一个方法的任何参数,那么第一个方法就相对不合适。比如上面的代码: 任何传递给f1(double[])方法的参数都可以传递给f1(Object)方法,而反之却不行,那么f1(double[])方法就更合适。因此JVM就会调用这个更合适的方法。
★ 总结
(1) 所有私有方法、静态方法、构造器及初始化方法<clinit>都是采用静态绑定机制。在编译器阶段就已经指明了调用方法在常量池中的符号引用,JVM运行的时候只需要进行一次常量池解析即可。
(2) 类对象方法的调用必须在运行过程中采用动态绑定机制。
首先,根据对象的声明类型(对象引用的类型)找到“合适”的方法。具体步骤如下:
① 如果能在声明类型中匹配到方法签名完全一样(参数类型一致)的方法,那么这个方法是最合适的。
② 在第①条不能满足的情况下,寻找可以“凑合”的方法。标准就是通过将参数类型进行自动转型之后再进行匹配。如果匹配到多个自动转型后的方法签名f(A)和f(B),则用下面的标准来确定合适的方法:传递给f(A)方法的参数都可以传递给f(B),则f(A)最合适。反之f(B)最合适 。
③ 如果仍然在声明类型中找不到“合适”的方法,则编译阶段就无法通过。
然后,根据在堆中创建对象的实际类型找到对应的方法表,从中确定具体的方法在内存中的位置。
★ 覆写(override)
一个实例方法可以覆写(override)在其超类中可访问到的具有相同签名的所有实例方法,从而使能了动态分派(dynamic dispatch);换句话说,VM将基于实例的运行期类型来选择要调用的覆写方法。覆写是面向对象编程技术的基础,并且是唯一没有被普遍劝阻的名字重用形式:
Java代码
1. class Base{
2. public void f(){}
3. }
4. class Derived extends Base{
5. public void f(){}
6. }
★ 隐藏(hide)
一个域、静态方法或成员类型可以分别隐藏(hide)在其超类中可访问到的具有相同名字(对方法而言就是相同的方法签名)的所有域、静态方法或成员类型。隐藏一个成员将阻止其被继承。
Java代码
1. class Base{
2. public static void f(){}
3. }
4. class Derived extends Base {
5. private static void f(){} //hides Base. f()
6. }
★ 重载(overload)
在某个类中的方法可以重载(overload)另一个方法,只要它们具有相同的名字和不同的签名。由调用所指定的重载方法是在编译期选定的。
Java代码
1. class CircuitBreaker{
2. public void f (int i){} //int overloading
3. public void f(String s){} //String overloading
4. }
★ 遮蔽(shadow)
一个变量、方法或类型可以分别遮蔽(shadow)在一个闭合的文本范围内的具有相同名字的所有变量、方法或类型。如果一个实体被遮蔽了,那么你用它的简单名是无法引用到它的;根据实体的不同,有时你根本就无法引用到它。
Java代码
1. class WhoKnows{
2. static String sentence=”I don't know.”;
3. public static void main(String[] args〕{
4. String sentence=”I don't know.”; //shadows static field
5. System.out. println (sentence); // prints local variable
6. }
7. }
尽管遮蔽通常是被劝阻的,但是有一种通用的惯用法确实涉及遮蔽。构造器经常将来自其所在类的某个域名重用为一个参数,以传递这个命名域的值。这种惯用法并不是没有风险,但是大多数Java程序员都认为这种风格带来的实惠要超过
其风险:
Java代码
1. class Belt{
2. private find int size ; //Parameter shadows Belt. size
3. public Belt (int size){
4. this. size=size;
5. }
6. }
★ 遮掩(obscure)
一个变量可以遮掩具有相同名字的一个类型,只要它们都在同一个范围内:如果这个名字被用于变量与类型都被许可的范围,那么它将引用到变量上。相似地,一个变量或一个类型可以遮掩一个包。遮掩是唯一一种两个名字位于不同的名字空间的名字重用形式,这些名字空间包括:变量、包、方法或类型。如果一个类型或一个包被遮掩了,那么你不能通过其简单
名引用到它,除非是在这样一个上下文环境中,即语法只允许在其名字空间中出现一种名字。遵守命名习惯就可以极大地消除产生遮掩的可能性:
Java代码
1. public class Obscure{
2. static String System;// Obscures type java.lang.System
3. public static void main(String[] args)
4. // Next line won't compile:System refers to static field
5. System. out. println(“hello, obscure world!”);
6. }
7. }
分享到:
相关推荐
总结起来,Java的同步机制底层实现主要依赖于`AbstractQueuedSynchronizer`,它为多线程同步提供了灵活且高效的框架。`ReentrantLock`和`CountDownLatch`是基于AQS的典型应用,通过它们我们可以深入理解Java并发编程...
### Java Core及底层面试问题概览 #### 一、JAVA面向对象 面向对象是Java语言的核心特性之一,它强调的是将复杂的问题抽象成简单的对象,然后通过这些对象之间的交互来解决问题。 1. **封装**: 封装是面向对象的...
这份“25本Java高手合集”无疑是Java开发者们的一份宝贵资源,它涵盖了各种底层分析和精华知识点,旨在帮助读者深入理解Java的核心原理,提升编程技能。 在Java的世界里,JDBC(Java Database Connectivity)是连接...
在Java编程语言的世界里,深入了解其底层机制是提升开发能力和优化代码的关键。本文将带你深入探索Java的包管理、类加载器以及import等核心概念,帮助你更好地理解Java运行时环境的工作原理。 1. **包管理**: ...
Java面试中的底层原理问题涵盖了许多核心概念,这些概念对于理解和优化Java应用程序至关重要。以下是对这些知识点的详细解释: 1. **HashSet**: - `HashSet` 是基于`HashMap`实现的,这意味着它利用了散列技术来...
本文将深入探讨Java并发的底层实现,帮助开发者更好地理解和应用并发技术。 首先,我们要明确并发编程的初衷是为了提升程序的运行效率,特别是在多核CPU环境下,通过合理地分配任务到多个线程,可以充分利用计算...
同时,注意Java的垃圾收集机制可能会影响到C++中的内存管理,确保两者之间的内存分配和释放协调一致。 8. **性能优化**: 虽然JNI允许Java调用C++,但是跨语言调用会有一定的性能开销。尽量减少不必要的调用,将复杂...
Java虚拟机(JVM)在创建和管理对象时涉及多个关键概念和技术,这些概念与对象的内存布局、对象头、对象锁以及`synchronized`关键字的底层实现密切相关。在JVM中,对象的创建过程分为几个步骤: 1. **类加载检查**:...
《Java实现的JVM——深入...总结,"jvmjava"项目是一个极好的学习资源,它使开发者有机会亲手构建一个JVM,从底层理解Java应用程序的运行机制。无论是初学者还是经验丰富的开发者,都能从中受益,提升自己的技术水平。
动态内存分配是计算机科学中的一个重要概念,特别是在C和C++等编程语言中,它允许程序在运行时根据需要请求和释放内存。...通过分析和实现这些算法,我们可以更好地理解内存管理的底层机制,并在实践中提高程序性能。
通过这样的模拟实践,开发者可以深入理解内存分配的底层机制,这对于优化代码、避免内存泄漏、提高应用性能具有重要意义。同时,对Java内存管理有深入理解的开发者能够编写出更加高效和健壮的程序。
在Java虚拟机中,本地方法接口(JNI,Java Native Interface)允许Java代码调用原生的C或C++代码,为Java提供了与操作系统底层交互的能力。本地方法支持使得JVM可以处理那些无法纯用Java实现的功能,如硬件操作、...
Java集合框架是存储和操作数据的基础,本书会剖析ArrayList、LinkedList、HashSet、HashMap等常用集合类的实现原理,并讨论它们在不同场景下的选择与优化。 异常处理是Java程序中的重要组成部分,它关乎程序的健壮...
### 基于Java多线程的并发机制的研究和实现 #### 1. 进程与线程的概念比较 在探讨Java多线程机制之前,我们先了解进程和线程的基本概念及其区别。 - **进程**:进程是操作系统中能够独立运行的基本单位,它包含了...
Java访问权限控制,为Java操作文件、写入文件分配合适的权限,定义写到文件的信息、定义文件,输出到c:/hello.txt、写信息到文件、关闭输出流。 Java绘制图片火焰效果 1个目标文件 摘要:Java源码,图形操作,火焰...
标题 "cpp-JNI封装的底层加密实现可跨平台使用" 提及的是在C++中使用JNI(Java Native Interface)来封装底层加密算法,使得这些加密功能可以在不同的操作系统平台上通用。JNI是Java平台的一个核心特性,它允许Java...
Java的类加载机制是其动态性的体现。当程序启动或类被首次引用时,JVM会将对应的类加载到内存中。类加载器负责查找和加载类的字节码文件,然后将其转换为可供JVM执行的格式。Java的类加载机制支持层次结构,包括引导...
Java虚拟机(JVM)是实现Java程序跨平台运行的关键技术,它的实现原理和工作过程是Java语言能够运行在各种不同硬件平台的基础。JVM的存在使得Java程序员可以编写一次代码,到处运行,这得益于JVM提供的一套平台无关...
C语言实现通常更为底层,更注重性能,而Java实现则更加面向对象,代码结构清晰,易于维护。在C实现中,你需要关注内存管理,比如动态分配和释放,以及位操作等细节。而在Java实现中,可以利用内置的数据结构和类库,...
转换器需确保所有动态分配的内存都被正确地释放,或者在Java中用适当的构造和析构函数替代。 6. **线程和并发**:C++和Java都有各自的线程模型,但API和语义有所不同。转换器需要考虑线程安全性和并发控制的转换。 ...