Java中方法调用分析!详细解析静态分派和动态分派的执行过程

简介: 本篇文章中介绍了Java中的方法调用以及方法的解析和执行过程。分析了在Java中方法的静态分派和动态分派的执行过程。最后分析了在虚拟机中静态分派和动态分派的实现方式。通过这篇文章的学习,可以帮助我们很好地认识Java中的方法的执行情况。

方法调用

  • 在程序运行时,进行方法调用是最普遍,最频繁的操作
  • 方法调用不等于方法执行:

    • 方法调用阶段唯一的任务就是确定被调用的方法版本,即调用哪一个方法
    • 不涉及方法内部的具体运行过程
  • Class文件的编译过程不包括传统编译中的连接步骤
  • Class文件中的一切方法调用在Class文件里面存储的都是符号引用,而不是方法在在实际运行时内存布局中的入口地址,即之前的直接引用:

    • 这样使得Java具有更强大的动态扩展能力
    • 同时也使得Java方法调用过程变得相对复杂
    • 需要在类加载期间,甚至会到运行期间才能确定目标方法的直接引用

方法解析

  • 所有方法调用中的目标方法在Class文件里都是一个常量池的引用
  • 在类的加载解析阶段,会将其中的一部分符号引用转化为直接引用:

    • 方法在程序真正执行之前就有一个可确定的调用版本,并且这个方法的调用版本在运行期是不可改变的
    • 也就是说,调用目标在程序代码中完成,编译器进行编译时就必须确定下来,这也叫做方法解析

Java方法分类

  • 在Java中符合 "编译期可知,运行期不可变" 的方法有两大类:

    • 静态方法: 与类型直接关联
    • 私有方法: 在外部不可被访问
    • 这两种方法各自的特点决定这两种方法都不可能通过继承或者别的方式重写版本,因此适合在类加载阶段进行解析
  • 非虚方法: 在类加载阶段会把符号引用解析为该方法的直接引用

    • 静态方法
    • 私有方法
    • 实例构造器
    • 父类方法
  • 虚方法: 在类加载阶段不会将符号引用解析为该方法的直接引用

    • 除去以上的非虚方法,其它的方法均为虚方法

静态分派

public class StaticDispatch {
    static abstract class Human {
    }
    
    static class Man extends Human {
    }
    static class Woman extends Human {
    }

    public static void sayHello(Human guy) {
        System.out.println("Hello, Guy!");
    }
    public static void sayHello(Man guy) {
        System.out.println("Hello, Gentleman!");
    }
    public static void sayHello(woman guy) {
        System.out.println("Hello, Lady!");
    }

    public static void main(String[] args) {
        Human man = new Man();
        Human women = new Woman();
        
        sayHello(man);
        sayHello(woman);
         
    }
}
Human man = new Human();
  • Human为变量的静态类型
  • Man为变量的实际类型
  • 静态类型和实际类型在程序中都会放生变化:

    • 静态类型:

      • 静态类型的变化仅仅在使用时发生
      • 变量本身的静态类型不会被改变
      • 最终的静态类型在编译器中可知
    • 实际类型:

      • 实际类型变化的结果在运行期才确定下来
      • 编译器在编译期间并不知道一个对象的实际类型是什么
Human human = new Man();
sayHello(man);
sayHello((Man)man);        // 类型转换,静态类型变化,转型后的静态类型一定是Man
man = new woman();        // 实际类型变化,实际类型是不确定的
sayHello(man);
sayHello((Woman)man);    // 类型转换,静态类型变化
  • 编译器在重载时是通过参数的静态类型而不是实际类型作为判断依据,静态类型在编译期间可以知道:

    • 编译阶段,Javac编译器会根据参数的静态类型决定使用哪个重载版本
  • 静态分派:

    • 所有依赖静态类型来定位方法的执行版本的分派动作
    • 典型应用 :方法重载
  • 静态分派发生在编译阶段,因此确定静态分派的的动作不是由虚拟机执行的,而是由编译器完成的
  • 由于字面量没有显示静态类型,只能通过语言上的规则去理解和推断
public class LiteralTest {
    public static void sayHello(char arg) {
        System.out.println("Hello, char!");
    }
    public static void sayHello(int arg) {
        System.out.println("Hello, int!");
    }
    public static void sayHello(long arg) {
        System.out.println("Hello, long!");
    }
    public static void sayHello(Character arg) {
        System.out.println("Hello, Character!");
    }
    public static void main(String[] arg) {
        sayHello('a');
    }
}
  • 编译器将重载方法从上向下依次注释,得到不同的输出
  • 如果编译器无法确定要自定转型为哪种类型,会提示类型模糊,拒绝编译
public class LiteralTest {
    public static void sayHello(String arg) {    // 新增重载方法
        System.out.println("Hello, String!");
    }
    public static void sayHello(char arg) {    
        System.out.println("Hello, char!");
    }
    public static void sayHello(int arg) {
        System.out.println("Hello, int!");
    }
    public static void sayHello(long arg) {
        System.out.println("Hello, long!");
    }
    public static void sayHello(Character arg) {
        System.out.println("Hello, Character!");
    }
    public static void main(String[] args) {
        Random r = new Random();
        String s = "abc";
        int i = 0;
        sayHello(r.nextInt() % 2 != 0 ? s : 1 );    // 编译错误
        sayHello(r.nextInt() % 2 != 0 ? 'a' : false);    //编译错误
    }
}

动态分派

public class DynamicDispatch {
    static abstract class Human {
        protected abstract void sayHello();
    }
    
    static class Man extends Human {
        @override
        protected void sayHello() {
            System.out.println("Man Say Hello!");
        }
    }
    static class Woman extends Human {
        @override
        protected void sayHello() {
            System.out.println("Woman Say Hello!");
        }
    }
    
    public static void main(String[] args) {
        Human man = new Man();
        Human women = new Woman();
        man.sayHello();
        woman.sayHello();
        man = new Woman();
        man.sayHello();
    }
}
  • 这里不是根据静态类型决定的

    • 静态类型的Human两个变量manwoman在调用sayHello() 方法时执行了不同的行为
    • 变量man在两次调用中执行了不同的方法
  • 导致这个现象的额原因 :这两个变量的实际类型不同
  • Java虚拟机是如何根据实际类型分派方法的执行版本的:invokevirtual指令的多态查找过程开始 ,invokevirtual指令运行时解析过程大致分为以下几个步骤:

    • 找到操作数栈顶的第一个元素所指向的对象的实际类型,记作C
    • 如果在类型C中找到与常量中的描述符和简单名称相符合的方法,然后进行访问权限验证,如果验证通过则返回这个方法的直接引用,查找过程结束;如果验证不通过,则抛出java.lang.illegalAccessError异常
    • 如果未找到,就按照继承关系从下往上依次对类型C的各个父类进行第二步的搜索和验证过程
    • 如果始终没有找到合适的方法,则抛出java.lang.AbstractMethodError异常
  • Java语言方法重写的本质:

    • invokevirtual指令执行的第一步就是在运行时期确定接收者的实际类型,所以两次调用中的invokevirtual指令把常量池中的类方法符号引用解析到了不同的直接引用上
  • 这种在运行时期根据实际类型确定方法执行版本的分派过程就叫做动态分派

虚拟机动态分派的实现

  • 虚拟机概念解析的模式就是静态分派和动态分派,可以理解虚拟机在分派中 "会做什么" 这个问题
  • 虚拟机 "具体是如何做到的" 在各种虚拟机实现上会有差别:

    • 由于动态分派是非常频繁的动作,而且动态分派的方法版本选择过程需要运行时在类的方法元数据中搜索合适的目标方法

      • 因此在虚拟机的实际实现中,为了基于性能的考虑,大部分实现都不会真正的进行如此频繁的搜索
      • 最常用的"稳定优化"的方式是为类在方法区中建立一个虚方法表(Virtual Method Table,即vtable), 使用虚方法表索引代替元数据查找以提高性能

        • 虚方法表中存放着各个方法的实际入口地址:

          • 如果某个方法在子类中没有被重写,那子类的虚方法表里面的地址入口和父类相同方法的地址入口是一致的,都指向父类的实际入口
          • 如果子类中重写了这个方法,子类方法表中的地址将会替换为指向子类实际方法的入口地址
        • 具有相同签名的方法,在父类,子类的虚方法表中具有一样的索引序号:

          • 这样当类型变换时,仅仅需要变更查找的方法表,就可以从不同的虚方法表中按索引转换出所需要的入口地址
        • 方法表一般在类加载阶段的连接阶段进行初始化:

          • 准备了类的变量初始值后,虚拟机会把该类的方法表也初始化完毕
相关文章
|
8月前
|
Java 开发者
重学Java基础篇—Java类加载顺序深度解析
本文全面解析Java类的生命周期与加载顺序,涵盖从加载到卸载的七个阶段,并深入探讨初始化阶段的执行规则。通过单类、继承体系的实例分析,明确静态与实例初始化的顺序。同时,列举六种触发初始化的场景及特殊场景处理(如接口初始化)。提供类加载完整流程图与记忆口诀,助于理解复杂初始化逻辑。此外,针对空指针异常等问题提出排查方案,并给出最佳实践建议,帮助开发者优化程序设计、定位BUG及理解框架机制。最后扩展讲解类加载器层次与双亲委派机制,为深入研究奠定基础。
314 0
|
8月前
|
数据采集 前端开发 JavaScript
金融数据分析:解析JavaScript渲染的隐藏表格
本文详解了如何使用Python与Selenium结合代理IP技术,从金融网站(如东方财富网)抓取由JavaScript渲染的隐藏表格数据。内容涵盖环境搭建、代理配置、模拟用户行为、数据解析与分析等关键步骤。通过设置Cookie和User-Agent,突破反爬机制;借助Selenium等待页面渲染,精准定位动态数据。同时,提供了常见错误解决方案及延伸练习,帮助读者掌握金融数据采集的核心技能,为投资决策提供支持。注意规避动态加载、代理验证及元素定位等潜在陷阱,确保数据抓取高效稳定。
260 17
|
8月前
|
存储 设计模式 Java
重学Java基础篇—ThreadLocal深度解析与最佳实践
ThreadLocal 是一种实现线程隔离的机制,为每个线程创建独立变量副本,适用于数据库连接管理、用户会话信息存储等场景。
286 5
|
8月前
|
存储 监控 安全
重学Java基础篇—类的生命周期深度解析
本文全面解析了Java类的生命周期,涵盖加载、验证、准备、解析、初始化、使用及卸载七个关键阶段。通过分阶段执行机制详解(如加载阶段的触发条件与技术实现),结合方法调用机制、内存回收保护等使用阶段特性,以及卸载条件和特殊场景处理,帮助开发者深入理解JVM运作原理。同时,文章探讨了性能优化建议、典型异常处理及新一代JVM特性(如元空间与模块化系统)。总结中强调安全优先、延迟加载与动态扩展的设计思想,并提供开发建议与进阶方向,助力解决性能调优、内存泄漏排查及框架设计等问题。
369 5
|
8月前
|
机器学习/深度学习 人工智能 Java
Java机器学习实战:基于DJL框架的手写数字识别全解析
在人工智能蓬勃发展的今天,Python凭借丰富的生态库(如TensorFlow、PyTorch)成为AI开发的首选语言。但Java作为企业级应用的基石,其在生产环境部署、性能优化和工程化方面的优势不容忽视。DJL(Deep Java Library)的出现完美填补了Java在深度学习领域的空白,它提供了一套统一的API,允许开发者无缝对接主流深度学习框架,将AI模型高效部署到Java生态中。本文将通过手写数字识别的完整流程,深入解析DJL框架的核心机制与应用实践。
494 3
|
8月前
|
安全 IDE Java
重学Java基础篇—Java Object类常用方法深度解析
Java中,Object类作为所有类的超类,提供了多个核心方法以支持对象的基本行为。其中,`toString()`用于对象的字符串表示,重写时应包含关键信息;`equals()`与`hashCode()`需成对重写,确保对象等价判断的一致性;`getClass()`用于运行时类型识别;`clone()`实现对象复制,需区分浅拷贝与深拷贝;`wait()/notify()`支持线程协作。此外,`finalize()`已过时,建议使用更安全的资源管理方式。合理运用这些方法,并遵循最佳实践,可提升代码质量与健壮性。
252 1
|
8月前
|
传感器 监控 Java
Java代码结构解析:类、方法、主函数(1分钟解剖室)
### Java代码结构简介 掌握Java代码结构如同拥有程序世界的建筑蓝图,类、方法和主函数构成“黄金三角”。类是独立的容器,承载成员变量和方法;方法实现特定功能,参数控制输入环境;主函数是程序入口。常见错误包括类名与文件名不匹配、忘记static修饰符和花括号未闭合。通过实战案例学习电商系统、游戏角色控制和物联网设备监控,理解类的作用、方法类型和主函数任务,避免典型错误,逐步提升编程能力。 **脑图速记法**:类如太空站,方法即舱段;main是发射台,static不能换;文件名对仗,括号要成双;参数是坐标,void不返航。
337 5
|
9月前
|
Java API 数据处理
深潜数据海洋:Java文件读写全面解析与实战指南
通过本文的详细解析与实战示例,您可以系统地掌握Java中各种文件读写操作,从基本的读写到高效的NIO操作,再到文件复制、移动和删除。希望这些内容能够帮助您在实际项目中处理文件数据,提高开发效率和代码质量。
240 4
|
9月前
|
XML JSON Java
Java中Log级别和解析
日志级别定义了日志信息的重要程度,从低到高依次为:TRACE(详细调试)、DEBUG(开发调试)、INFO(一般信息)、WARN(潜在问题)、ERROR(错误信息)和FATAL(严重错误)。开发人员可根据需要设置不同的日志级别,以控制日志输出量,避免影响性能或干扰问题排查。日志框架如Log4j 2由Logger、Appender和Layout组成,通过配置文件指定日志级别、输出目标和格式。
|
10月前
|
存储 Java 计算机视觉
Java二维数组的使用技巧与实例解析
本文详细介绍了Java中二维数组的使用方法
355 15