深入分析Java虚拟机中方法执行流程及方法重载和方法重写原理

  • 前言
  • 思考
  • 栈帧
    • 局部变量表(Local Variables)
    • 操作数栈(Operand Stacks)
    • 动态连接(Dynamic Linking)
    • 方法返回地址
      • 正常退出(Normal Method Invocation Completion)
      • 异常终止(Abrupt Method Invocation Completion)
    • 其他附加信息
  • 方法调用流程演示
  • 方法调用分析
    • 方法调用指令
    • 方法解析
    • 非虚方法
    • 方法重载
    • 宗量
    • 静态分派
    • 方法重写
    • 动态分派
    • 单分派与多分派
  • 总结

前言

JVM执行字节码指令是基于栈的架构,就是说所有的操作数都必须先入栈,然后再根据需要出栈进行操作计算,再把结果进行入栈,这个流程和基于寄存器的架构是有本质区别的,而基于寄存器架构来实现,在不同的机器上可能会无法做到完全兼容,这也是Java会选择基于栈的设计的原因之一。

思考

我们思考下,当我们调用一个方法时,参数是怎么传递的,返回值又是怎么保存的,一个方法调用之后又是如何继续下一个方法调用的呢?调用过程中肯定会存储一些方法的参数和返回值等信息,这些信息存储在哪里呢?
JVM系列文章1中我们提到了,每次调用一个方法就会产生一个栈帧,所以我们肯定可以想到栈帧就存储了所有调用过程中需要使用到的数据。现在就让我们深入的去了解一下Java虚拟机栈中的栈帧吧。

栈帧

当我们调用一个方法的时候,就会产生一个栈帧,当一个方法调用完成时,它所对应的栈帧将被销毁,无论这种完成是正常的还是突然的(抛出一个未捕获的异常)。

每个栈帧中包括局部变量表(Local Variables)、操作数栈(Operand Stack)、动态链接(Dynamic Linking)、方法返回地址(Return Address)和额外的附加信息。

在给定的线程当中,永远只有一个栈帧是活动的,所以活动的栈帧又称之为当前栈帧,而其对应的方法则称之为当前方法,定义了当前方法的类则称之为当前类。当一个方法调用结束时,其对应的栈帧也会被丢弃。

局部变量表(Local Variables)

局部变量表是以数组的形式存储的,而且当前栈帧的方法所需要分配的最大长度是在编译时就确定了。局部变量表通过index来寻址,变量从index[0]开始传递。

局部变量表的数组中,每一个位置可以保存一个32位的数据类型:boolean、byte、char、short、int、float、reference或returnAddress类型的值。而对于64位的数据类型long和double则需要两个位置来存储,但是因为局部变量表是属于线程私有的,所以虽然被分割为2个变量存储,依然不用担心会出现安全性问题。

对于64位的数据类型,假如其占用了数组中的index[n]和index[n+1]两个位置,那么不允许单独访问其中的某一个位置,Java虚拟机规范中规定,如果出现一个64位的数据被单独访问某一部分时,则在类加载机制中的校验阶段就应该抛出异常。

Java虚拟机在方法调用时使用局部变量进行传递参数。在类方法(static方法)调用中,所有参数都以从局部变量中的index[0]开始进行参数传递。而在实例方法调用上,index[0]固定用来传递方法所属于的对象实例,其余所有参数则在从局部变量表内index[1]的位置开始进行传递。

注意:局部变量表中的变量不可以直接使用,如需要使用的话,必须通过相关指令将其加载至操作数栈中作为操作数才能使用

操作数栈(Operand Stacks)

操作数栈,在上下文语义清晰时,也可以称之为操作栈(Operand Stack),是一个后进先出(Last In First Out,LIFO)栈,同局部变量表一样,操作数栈的最大深度也是在编译时就确定的。

操作数栈在刚被创建时(也就是方法刚被执行的时候)是空的,然后在执行方法的过程中,通过虚拟机指令将常量/值从局部变量表或字段加载到操作数栈中,然后对其进行操作,并将操作结果压入栈内。

操作数堆栈上的每个条目都可以保存任何Java虚拟机类型的值,包括long或double类型的值。

注意:我们必须以适合其类型的方式对操作数堆栈中的值进行操作。例如,不可能将两个int类型的值压入栈后将其视为long类型,也不可能将两个float类型值压入栈内后使用iadd指令将其添加

动态连接(Dynamic Linking)

每个栈帧都包含一个指向运行时常量池中该栈帧所属方法的引用,持有这个引用是为了支持方法调用过程中的动态连接。

在Class文件中的常量池中存有大量的符号引用,字节码中的方法调用指令就以常量池中指向方法的符号引用作为参数,这些符号引用一部分会在类加载阶段或者第一次使用的时候就转化为直接引用,这种就称为静态解析。而另外一部分则会在每一次运行期间才会转化为直接引用,这部分就称为动态连接。

方法返回地址

当一个方法开始执行后,只有两种方式可以退出:一种是遇到方法返回的字节码指令;一种是遇见异常,并且这个异常没有在方法体内得到处理。

正常退出(Normal Method Invocation Completion)

如果对当前方法的调用正常完成,则可能会向调用方法返回一个值。当被调用的方法执行其中一个返回指令时,返回指令的选择必须与被返回值的类型相匹配(如果有的话)。

方法正常退出时,当前栈帧通过将调用者的pc程序计数器适当的并跳过当前的调用指令来恢复调用程序的状态,包括它的局部变量表和操作数堆栈。然后继续在调用方法的栈帧来执行后续流程,如果有返回值的话则需要将返回值压入操作数栈。

异常终止(Abrupt Method Invocation Completion)

如果在方法中执行Java虚拟机指令导致Java虚拟机抛出异常,并且该异常没有在方法中处理,那么方法调用会突然结束,因为异常导致的方法突然结束永远不会有返回值返回给它的调用者。

其他附加信息

这一部分具体要看虚拟机产商是如何实现的,虚拟机规范并没有对这部分进行描述。

方法调用流程演示

上面的概念听起来有点抽象,下面我们就通过一个简单的例子来演示一下方法的执行流程。

package com.zwx.jvm;public class JVMDemo {public static void main(String[] args) {int sum = add(1, 2);print(sum);}public static int add(int a, int b) {a = 3;int result = a + b;return result;}public static void print(int num) {System.out.println(num);}
}

要想了解Java虚拟机的执行流程,那么我们必须要对类进行编译,得到字节码文件,执行如下命令:

javap -c xxx\xxx\JVMDemo.class >1.txt

将JVMDemo.class生成的字节码指令输出到1.txt文件中,然后打开,看到如下字节码指令:

Compiled from "JVMDemo.java"
public class com.zwx.jvm.JVMDemo {public com.zwx.jvm.JVMDemo();Code:0: aload_01: invokespecial #1                  // Method java/lang/Object."<init>":()V4: returnpublic static void main(java.lang.String[]);Code:0: iconst_11: iconst_22: invokestatic  #2                  // Method add:(II)I5: istore_16: iload_17: invokestatic  #3                  // Method print:(I)V10: returnpublic static int add(int, int);Code:0: iconst_31: istore_02: iload_03: iload_14: iadd5: istore_26: iload_27: ireturnpublic static void print(int);Code:0: getstatic     #4                  // Field java/lang/System.out:Ljava/io/PrintStream;3: iload_04: invokevirtual #5                  // Method java/io/PrintStream.println:(I)V7: return
}

如果是第一次接触可能指令看不太懂,但是大致的类结构还是很清晰的,我们先来对用到的字节码指令大致说明一下:

  • iconst_i
    表示将整型数字i压入操作数栈,注意,这里i的返回只有-1~5,如果不在这个范围会采用其他指令,如当int取值范围是[-128,127]时,会采用bipush指令。
  • invokestatic
    表示调用一个静态方法
  • istore_n
    这里表示将一个整型数字存入局部变量表的索引n位置,因为局部变量表是通过一个数组形式来存储变量的
  • iload_n
    表示将局部变量位置n的变量压入操作数栈
  • ireturn
    将当前方法的结果返回到上一个栈帧
  • invokevirtual
    调用虚方法

了解了字节码指令的大概意思,接下来就让我们来演示一下主要的几个执行流程:

  • 1、代码编译之后大致得到如下的一个Java虚拟机栈,注意这时候操作数栈都是空的(pc寄存器的值在这里暂不考虑 ,实际上调用指令的过程,pc寄存器是会一直发生变化的)

  • 2、执行iconst_1和iconst_2两个指令,也就是从本地变量中把整型1和2两个数字压入操作数栈内:

  • 3、执行invokestatic指令,调用add方法,会再次创建一个新的栈帧入栈,并且会将参数a和b存入add栈帧中的本地变量表

  • 4、add栈帧中调用iconst_3指令,从本地变量中将整型3压入操作数栈

  • 5、add栈帧中调用istore_0,表示将当前的栈顶元素存入局部变量表index[0]的位置,也就是赋值给a。

  • 6、调用iload_0和iload_1,将局部变量表中index[0]和index[1]两个位置的变量压入操作数栈

  • 7、最后执行iadd指令:将3和2弹出栈后将两个数相加,得到5,并将得到的结果5重新压入栈内

    8、执行istore_2指令,将当前栈顶元素弹出存入局部变量表index[2]的位置,并再次调用iload_2从局部变量表内将index[2]位置的数据压入操作数栈内

  • 9、最后执行ireturn命令将结果5返回main栈帧,此时栈帧add被销毁,回到main栈帧继续后续执行

    方法的调用大致就是不断的入栈和出栈的过程,上述的过程省略了很多细节,只关注了大致流程即可,实际调用比图中要复杂的多。

方法调用分析

我们知道,Java是一种面向对象语言,支持多态,而多态的体现形式就是方法重载和方法重写,那么Java虚拟机又是如何确认我们应该调用哪一个方法的呢?

方法调用指令

首先,我们来看一下方法的字节码调用指令,在Java中,提供了4种字节码指令来调用方法(jdk1.7之前):

  • 1、invokestatic:调用静态方法
  • 2、invokespecial:调用实例构造器方法,私有方法,父类方法
  • 3、invokevirtual:调用所有的虚方法
  • 4、invokeinterface:调用接口方法(运行时会确定一个实现了接口的对象)

注意:在JDK1.7开始,Java新增了一个指令invokedynamic,这个是为了实现“动态类型语言”而引入的,在这里我们暂不讨论

方法解析

在类加载机制中的解析阶段,主要做的事情就是将符号引用转为直接引用,但是,对方法的调用而言,有一个前提,那就是在方法真正运行之前就可以唯一确定具体要调用哪一个方法,而且这个方法在运行期间是不可变的。只有满足这个前提的方法才会在解析阶段直接被替换为直接引用,否则只能等到运行时才能最终确定。

非虚方法

在Java语言中,满足“编译器可知,运行期不可变”这个前提的方法,被称之为非虚方法。非虚方法在类加载机制中的解析阶段就可以直接将符号引用转化为直接引用。非虚方法有4种:

  • 1、静态方法
  • 2、私有方法
  • 3、实例构造器方法
  • 4、父类方法(通过super.xxx调用,因为Java是单继承,只有一个父类,所以可以确定方法的唯一)

除了非虚方法之外的非final方法就被称之为虚方法,虚方法需要运行时才能确定真正调用哪一个方法。Java语言规范中明确指出,final方法是一种非虚方法,但是final又属于比较特殊的存在,因为final方法和其他非虚方法调用的字节码指令不一样

知道了虚方法的类型,再结合上面的方法的调用指令,我们可以知道,虚方法就是通过字节码指令invokestatic和invokespecial调用的,而final方法又是一个例外,final方法是通过字节码指令invokevirtual调用的,但是因为final方法的特性就是不可被重写,无法覆盖,所以必然是唯一的,虽然调用指令不同,但是依然属于非虚方法的范畴。

方法重载

先来看一个方法重载的例子:

package com.zwx.jvm.overload;public class OverloadDemo {static class Human {}static class Man extends Human {}static class WoMan extends Human {}public void hello(Human human) {System.out.println("Hi,Human");}public void hello(Man man) {System.out.println("Hi,Man");}public void hello(WoMan woMan) {System.out.println("Hi,Women");}public static void main(String[] args) {OverloadDemo overloadDemo = new OverloadDemo();Human man = new Man();Human woman = new WoMan();overloadDemo.hello(man);overloadDemo.hello(woman);}
}

输出结果为:

Hi,Human
Hi,Human

这里,Java虚拟机为什么会选择参数为Human的方法来进行调用呢?

在解释这个问题之前,我们先来介绍一个概念:宗量

宗量

方法的接收者(调用者)和方法参数统称为宗量。而最终决定方法的分派就是基于宗量来选择的,故而根据基于多少种宗量来选择方法又可以分为:

  • 单分派:根据1个宗量对方法进行选择
  • 多分派:根据1个以上的宗量对方法进行选择

知道了方法的分派是基于宗量来进行的,那我们再回到上面的例子中就很好理解了。

overloadDemo.hello(man);

这句代码中overloadDemo表示接收者,man表示参数,而接收者是确定唯一的,就是overloadDemo实例,所以决定调用哪个方法的只有参数(包括参数类型和个数和顺序)这一个宗量。我们再看看参数类型:

Human man = new Man();

这句话中,Human称之为变量的静态类型,而Man则称之为变量的实际类型,而Java虚拟机在确认重载方法时是基于参数的静态类型来作为判断依据的,故而最终实际上不管你右边new的对象是哪个,调用的都是参数类型为Human的方法。

静态分派

所有依赖变量的静态类型来定位方法执行的分派动作就称之为静态分派。静态分派最典型的应用就是方法重载。

方法重载在编译期就能确定方法的唯一,不过虽然如此,但是在有些情况下,这个重载版本不是唯一的,甚至是有点模糊的。产生这个原因就是因为字面量并不需要定义,所以字面量就没有今天类型,比如我们直接调用一个方法:xxx.xxx(‘1’),这个字面量1就是模糊的,并没有对应静态类型。我们再来看一个例子:

package com.zwx.jvm.overload;import java.io.Serializable;public class OverloadDemo2 {public static void hello(Object a){System.out.println("Hello,Object");}public static void hello(double a){System.out.println("Hello,double");}public static void hello(Double a){System.out.println("Hello,Double");}public static void hello(float a){System.out.println("Hello,float");}public static void hello(long a){System.out.println("Hello,long");}public static void hello(int a){System.out.println("Hello,int");}public static void hello(Character a){System.out.println("Hello,Character");}public static void hello(char a){System.out.println("Hello,char");}public static void hello(char ...a){System.out.println("Hello,chars");}public static void hello(Serializable a){System.out.println("Hello,Serializable");}public static void main(String[] args) {OverloadDemo2.hello('1');}
}

这里的输出结果是

Hello,char

然后如果把该方法注释掉,就会输出:

Hello,int

再把int方法注释掉,那么会依次按照如下顺序进行方法调用输出:

char->int->long->float->double->Character->Serializable->Object->chars

可以看到,多参数的优先级最低,之所以会输出Serializable是因为包装类Character实现了Serializable接口,注意示例中double的包装类Double,并不会被执行。

方法重写

我们把上面第1个例子修改一下:

package com.zwx.jvm.override;public class OverrideDemo {static class Human {public void hello(Human human) {System.out.println("Hi,Human");}}static class Man extends Human {@Overridepublic void hello(Human human) {System.out.println("Hi,Man");}}static class WoMan extends Human {@Overridepublic void hello(Human human) {System.out.println("Hi,Women");}}public static void main(String[] args) {Human man = new Man();Human woman = new WoMan();man.hello(man);man.hello(woman);woman.hello(woman);woman.hello(man);}
}

输出结果为:

Hi,Man
Hi,Man
Hi,Women
Hi,Women

这里静态类型都是Human,但是却输出了两种结果,所以肯定不是按照静态类型来分派方法了,而从结果来看应该是按照了调用者的实际类型来进行的判断。

执行javap命令把类转换成字节码:

Compiled from "OverrideDemo.java"
public class com.zwx.jvm.override.OverrideDemo {public com.zwx.jvm.override.OverrideDemo();Code:0: aload_01: invokespecial #1                  // Method java/lang/Object."<init>":()V4: returnpublic static void main(java.lang.String[]);Code:0: new           #2                  // class com/zwx/jvm/override/OverrideDemo$Man3: dup4: invokespecial #3                  // Method com/zwx/jvm/override/OverrideDemo$Man."<init>":()V7: astore_18: new           #4                  // class com/zwx/jvm/override/OverrideDemo$WoMan11: dup12: invokespecial #5                  // Method com/zwx/jvm/override/OverrideDemo$WoMan."<init>":()V15: astore_216: aload_117: aload_118: invokevirtual #6                  // Method com/zwx/jvm/override/OverrideDemo$Human.hello:(Lcom/zwx/jvm/override/OverrideDemo$Human;)V21: aload_122: aload_223: invokevirtual #6                  // Method com/zwx/jvm/override/OverrideDemo$Human.hello:(Lcom/zwx/jvm/override/OverrideDemo$Human;)V26: aload_227: aload_228: invokevirtual #6                  // Method com/zwx/jvm/override/OverrideDemo$Human.hello:(Lcom/zwx/jvm/override/OverrideDemo$Human;)V31: aload_232: aload_133: invokevirtual #6                  // Method com/zwx/jvm/override/OverrideDemo$Human.hello:(Lcom/zwx/jvm/override/OverrideDemo$Human;)V36: return
}

我们可以发现这里的方法调用使用了指令invokevirtual来调用,因为根据上面的分类可以判断,hello方法均是虚方法

main方法大概解释一下,main方法中,第7行(Code列序号)和第15行是分别把Man对象实例和Women对象实例存入局部变量变的index[1]和index[2]两个位置,然后16,17两行,21,22两行,26,27两行,31,32两行分别是把需要用到的方法调用者和参数压入操作数栈,然后调用invokevirtual指令调用方法

所以上面最关键的就是invokevirtual指令到底是如何工作的呢?invokevirtual主要是按照如下步骤进行方法选择的:

  • 1、找到当前操作数栈中的方法接收者(调用者),记下来,比如叫Caller
  • 2、然后在类型Caller中去找方法,如果找到方法签名一致的方法,则停止搜索,开始对方法校验,校验通过直接调用,校验不通过,直接抛IllegalAccessError异常
  • 3、如果在Caller中没有找到方法签名一致的方法,则往上找父类,以此类推,直到找到为止,如果到顶了还没找到匹配的方法,则抛出AbstractMethodError异常

动态分派

上面的方法重写例子中,在运行期间才能根据实际类型来确定方法的执行版本的分派过程就称之为动态分派。

单分派与多分派

上面方法重载的第1个示例中,是一个静态分派过程,静态分配过程中Java虚拟机选择目标方法有两点:

  • 1、静态类型
  • 2、方法参数
    也就是用到了2个宗量来进行分派,所以是一个静态多分派的过程。

而上面方法重写的例子中,因为方法签名是固定的,也就是参数是固定的,那么就只有一个宗量-静态类型,能最终确定方法的调用,所以属于动态单分派。

所以可以得出对Java而言:Java是一门静态多分派,动态单分派语言

总结

本文主要介绍了一下Java虚拟机中,方法的执行流程以及方法执行过程中时,Java虚拟机栈中的内存布局,并从字节码的角度诠释了Java虚拟机是如何针对方法重载和方法重写来做出最终调用方法的选择的。

下一篇,将会介绍Java对象在内存中的布局,以及堆这种作为所有线程共享的的内存区域中具体又是如何存储对象的。
请关注我,和孤狼一起学习进步

【JVM系列3】方法重载和方法重写原理分析,看完这篇终于彻底搞懂了相关推荐

  1. JAVA SE、EE、ME,JRE、JDK,基本数据类型,访问修饰符、函数、封装的概念、UML类图、构造方法、this关键字、static关键字、方法重载、方法重写、包(packahe)

    运行第一个JAVA程序: 这里使用的开发环境是eclipse,新建一个java工程,然后可以看到src这个是存放java代码的地方,然后在src文件右击新建一个class(类),然后可以看到下图,同样 ...

  2. Java中方法重载和方法重写的区别

    文章目录 1 Java中方法重载和方法重写的区别 1 Java中方法重载和方法重写的区别 主要区别如下: 方法重载: 在同一个类中 方法名相同 参数个数.顺序.类型不同 返回值类型.访问修饰符任意 方 ...

  3. 语法基础(三. 类,属性,方法,方法重载,方法重写,构造方法,访问修饰符)

    语法基础(三. 类,属性,方法,方法重载,方法重写,构造方法,访问修饰符) (如有错误,欢迎指正,感谢!) 类 类是面向对象的程序设计中的概念,实现信息的封装 概念: 类就是拥有相等行为和相同的属性的 ...

  4. 令人发指的关于方法重载和方法重写的一些理解(多态)

    令人发指的关于方法重载和方法重写的一些理解 文章目录 **令人发指的关于方法重载和方法重写的一些理解** **方法重载的基础** **子类的方法重载** **重载升级只重写** **顺藤摸瓜之干掉&q ...

  5. java关于重写正确,在Java中,以下关于方法重载和方法重写描述正确的是( )

    在Java中,以下关于方法重载和方法重写描述正确的是( ) 答:方法重写的返回值类型必须相同或相容.(或是其子类) 对乙酰氨基酚临床用途是抗炎抗风湿作用 答:错 观念艺术是哲学的反思,传统的理论可以对 ...

  6. 方法重载和方法重写的区别和作用

    一.方法重载(Overload) 背景: 有时我们要用到一个函数同时兼容多种类型或者参数的情况,这时我们就要用到方法重载. 含义: 一个方法的逻辑是一样的,但是考虑到要适用不同的场景(参数个数不同,参 ...

  7. 教妹学Java:傻傻分不清,方法重载和方法重写

    方法重载和方法重写 01.开篇 入冬的夜,总是来得特别的早.我静静地站在阳台,目光所及之处,不过是若隐若现的钢筋混凝土,还有那毫无情调的灯光. "哥,别站在那发呆了.今天学啥啊,七点半我就要 ...

  8. [转载] 用大白话解释Java的方法重载和方法覆盖

    参考链接: Java中方法重载的不同方法 本文原创首发CSDN,本文链接https://blog.csdn.net/qq_41464123/article/details/107656852 ,作者博 ...

  9. JAVA 方法重载和方法覆写的区别(有代码)

    方法重载(override)与方法覆写(overload)的区别 1.方法名相同,但是各自的参数不同,叫做方法重载.方法重载的返回值都相同.其功能类的方法都使用同一名字. 举一个简单的方法重载的例子, ...

最新文章

  1. 数据中台应该包含什么?
  2. list修改元素的值_Python 到底是值传递还是引用传递
  3. Spring Boot微服务中Chaos Monkey的应用
  4. 硬盘最多能分几个区?
  5. Android开发过程中在sh,py,mk文件中添加log信息的方法
  6. 将DataFrame中的每一列分别做归一化处理的函数实现
  7. 【Python基础】这份 pip 使用小抄,要有全有多全(建议收藏)
  8. (转)iOS Wow体验 - 第五章 - 利用iOS技术特性打造最佳体验
  9. java进程调度怎么画图,[Java教程]进程调度的两种算法JAVA实现
  10. 【Kafka】kafka报错 UnknownHostException: %HOSTGROUP::host_group_zookeeper%: Temporary failure in name re
  11. cad插入块_CAD中的quot;块quot;能否修改?能!一个命令可快速解决
  12. 特斯拉起诉小鹏汽车员工窃取 30 多万份商业机密,何小鹏回应
  13. Win11怎么把右键刷新调出来?
  14. PLSQL Developer 64位 安装方法
  15. 自动化测试验证码代码常用的四种方式
  16. R语言混合线性模型包代码演示
  17. 色度抽样(4:2:0)到底是什么意思?
  18. 获取设备Mac地址和IP地址
  19. 人在当时处境中,像旋涡中的一片落叶,身不由己
  20. JavaScript 实现网页截屏五种方法

热门文章

  1. 北京上市公司招聘.net架构师及开发人员
  2. java集合面试锦集
  3. 安装win8的坎坷经历
  4. 百度云盘APP中去除我的应用数据图标:ES File Exploer
  5. 【干货】洋葱淘elya妞:电商小白产品操盘心得,洋葱淘如何搞到种子用户
  6. 命令行 修复系统_让我们修复旧的命令行
  7. 协方差意味着什么_微服务意味着我们可以使用所需的任何语言? 真?
  8. PDF怎么压缩指定大小
  9. 数据模拟:利用Java模拟数据(姓名,邮箱,地址,电话等信息,时间,工资,1-10随机数)并存入mysql
  10. mybatis一级缓存和二级缓存数据脏读问题