Java多态的本质—动态分派

简介: Java多态的本质—动态分派

概述


Java语言的一大特性是多态性,所谓多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量倒底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在由程序运行期间才能决定。

举个简单的例子,比如Human human = flag ? new Man() : new Woman(), human的具体类型是man还是woman在编写代码的时候我们是无法确定,它是由flag这个标记决定,只有在程序运行的时候才能够确定下来,这种让引用变量在运行时绑定到各种不同的类实现上,从而导致该引用调用的具体方法随之改变,即不修改程序代码就可以改变程序运行时所绑定的具体代码,让程序可以选择多个运行状态,这就是多态性。


例子


多态在Java中有两种实现形式,分别是继承和接口,子类重写父类或者接口中的方法,现在举个例子。

public class DynamicDispatch {
    static abstract class Animal {
        protected abstract void eat();
    }
    static class Cat extends Animal {
        @Override
        protected void eat() {
            System.out.println("我吃鱼");
        }
    }
    static class Dog extends Animal {
        @Override
        protected void eat() {
            System.out.println("我吃骨头");
        }
    }
    public static void main(String[] args) {
        Animal cat = new Cat();
        Animal dog = new Dog();
        cat.eat();
        dog.eat();
        cat = new Dog();
        cat.eat();
    }
}

运行结果:

1671176037291.jpg

这个结果相信和大家想的是一致的,那大家有想过JVM是怎么找到具体的类型执行的呢?我们定义的引用类型就是Animal,JVM是根据什么来找到对应的Cat 或者Dog这些具体的实例执行对应的方法呢?


从字节码角度分析


利用idea的Jclasslib插件查看字节码:

1671176046659.jpg

  1. 0~15行主要是创建Cat对象和Dog对象的字节码指令。
  2. 17和21行一模一样,指令都是invokevirtual, 参数都是<com/alvin/chapter8/DynamicDispatch$Animal.eat。竟然这两条指令一模一样,那他是怎么确定调用哪个实际类型的方法呢?这还得要了解invokevirtual指令的运行过程:
  • 找到操作数栈顶的第一个元素所指向的对象作为实际类型,记作类型C,这个是在运行期确定的。
  • 如果在类型C中找到与常量中的描述符和简单名称都相符的方法,则进行访问权限校验,通过返回这个方法的直接引用,查过过程结束。
  • 否则,按照继承关系从下往上依次对C的各个父类进行搜索和验证。
  • 如果始终没有找到合适的方法,抛出AbstractMethodError异常。
  1. 回过头来看,我们看到字节码中的第16行和20行的aload指令就是把刚刚创建的对象压入到栈顶。

以上的过程中根据方法接收者的实际类型来确定调用那个方法,找不到往父类继续找的过程,其实也就是重写的本质。我们把这种在运行期根据实际类型确定方法执行版本的分派过程叫做动态分派。


虚拟机动态分派的实现


上面讲述了虚拟即动态分派的过程,那它是怎么实现这一过程的呢?

因为动态分派是执行非常频繁的动作,而且需要在运行时搜索合适的目标方法,基于性能的考虑,java虚拟机采用了一种基础且常见的优化手段—为类型在方法区建立一个需方法表。使用需方法表索引来代替元数据查找以提高性能。

虚方法表中存放着各个方法的实际入口地址。如果某个方法在子类中没有被重写,那子类的虚方法表中的地址入口和父类相同方法的地址入口时一致的,如果子类重写了方法,子类虚方法表中的地址会被替换为指向子类实现版本的入口地址。

目录
相关文章
|
5月前
|
算法 Java 程序员
在Java的编程世界里,多态不仅仅是一种代码层面的技术,它是思想的碰撞,是程序员对现实世界复杂性的抽象映射,是对软件设计哲学的深刻领悟。
在Java的编程世界里,多态不仅仅是一种代码层面的技术,它是思想的碰撞,是程序员对现实世界复杂性的抽象映射,是对软件设计哲学的深刻领悟。
81 9
|
5月前
|
Java 开发者
在Java面向对象编程的广阔海洋中,多态犹如一股深邃的潜流,它推动着代码从单一走向多元,从僵化迈向灵活。
在Java面向对象编程的广阔海洋中,多态犹如一股深邃的潜流,它推动着代码从单一走向多元,从僵化迈向灵活。
45 7
|
5月前
|
Java 开发者
那些年,我们一同踏入Java编程的大门,多态,这个充满魔法的名字,曾无数次点亮我们探索面向对象编程的热情。
那些年,我们一同踏入Java编程的大门,多态,这个充满魔法的名字,曾无数次点亮我们探索面向对象编程的热情。
53 5
|
5月前
|
Java 程序员
让我们一起探讨Java多态的奥秘,看看它是如何打破“一刀切”的局限,让我们的代码更加生动多彩
让我们一起探讨Java多态的奥秘,看看它是如何打破“一刀切”的局限,让我们的代码更加生动多彩
46 5
|
5月前
|
Java 程序员
Java中的继承和多态:理解面向对象编程的核心概念
【8月更文挑战第22天】在Java的世界中,继承和多态不仅仅是编程技巧,它们是构建可维护、可扩展软件架构的基石。通过本文,我们将深入探讨这两个概念,并揭示它们如何共同作用于面向对象编程(OOP)的实践之中。你将了解继承如何简化代码重用,以及多态如何为程序提供灵活性和扩展性。让我们启程,探索Java语言中这些强大特性的秘密。
|
3月前
|
存储 Java 测试技术
Java零基础-多态详解
【10月更文挑战第10天】Java零基础教学篇,手把手实践教学!
41 4
|
3月前
|
Java 编译器 程序员
Java多态背后的秘密:动态绑定如何工作?
本文介绍了Java中多态的实现原理,通过动态绑定和虚拟方法表,使得父类引用可以调用子类的方法,增强了代码的灵活性和可维护性。文中通过具体示例详细解析了多态的工作机制。
79 4
|
4月前
|
Java 编译器
封装,继承,多态【Java面向对象知识回顾①】
本文回顾了Java面向对象编程的三大特性:封装、继承和多态。封装通过将数据和方法结合在类中并隐藏实现细节来保护对象状态,继承允许新类扩展现有类的功能,而多态则允许对象在不同情况下表现出不同的行为,这些特性共同提高了代码的复用性、扩展性和灵活性。
封装,继承,多态【Java面向对象知识回顾①】
|
3月前
|
Java
java继承和多态详解
java继承和多态详解
54 5
|
4月前
|
Java 编译器
Java——类与对象(继承和多态)
本文介绍了面向对象编程中的继承概念,包括如何避免重复代码、构造方法的调用规则、成员变量的访问以及权限修饰符的使用。文中详细解释了继承与组合的区别,并探讨了多态的概念,包括向上转型、向下转型和方法的重写。此外,还讨论了静态绑定和动态绑定的区别,以及多态带来的优势和弊端。
92 9
Java——类与对象(继承和多态)