iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么使用Java方法调用解析静态分派和动态分派
  • 943
分享到

怎么使用Java方法调用解析静态分派和动态分派

2023-07-02 12:07:11 943人浏览 薄情痞子
摘要

这篇“怎么使用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);// 类型转换,静态类型变化,转型后的静态类型一定是Manman = 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 {@overrideprotected void sayHello() {System.out.println("Man Say Hello!");}}static class Woman extends Human {@overrideprotected 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), 使用虚方法表索引代替元数据查找以提高性能

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

  • 如果某个方法在子类中没有被重写,那子类的虚方法表里面的地址入口和父类相同方法的地址入口是一致的,都指向父类的实际入口

  • 如果子类中重写了这个方法,子类方法表中的地址将会替换为指向子类实际方法的入口地址

具有相同签名的方法,在父类,子类的虚方法表中具有一样的索引序号:

这样当类型变换时,仅仅需要变更查找的方法表,就可以从不同的虚方法表中按索引转换出所需要的入口地址

方法表一般在类加载阶段的连接阶段进行初始化:

准备了类的变量初始值后,虚拟机会把该类的方法表也初始化完毕

以上就是关于“怎么使用Java方法调用解析静态分派和动态分派”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网精选频道。

--结束END--

本文标题: 怎么使用Java方法调用解析静态分派和动态分派

本文链接: https://www.lsjlt.com/news/341835.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 怎么使用Java方法调用解析静态分派和动态分派
    这篇“怎么使用Java方法调用解析静态分派和动态分派”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“怎么使用Java方法调用解...
    99+
    2023-07-02
  • Java方法调用解析静态分派动态分派执行过程
    方法调用 在程序运行时,进行方法调用是最普遍,最频繁的操作 方法调用不等于方法执行: 方法调用阶段唯一的任务就是确定被调用的方法版本,即调用哪一个方法不涉及方法内部的具体运行过程 C...
    99+
    2024-04-02
  • java JVM方法分派模型静态分派动态分派全面讲解
    目录前言1. 知识储备1.1 分派1.2 变量的静态类型 & 动态类型2. 静态分派a. 变量的静态类型 发生变化 的情况b. 静态分派的优先级匹配问题3. 动态分派4. 二...
    99+
    2024-04-02
  • JVM 方法调用之静态分派(详解)
    分派(Dispatch)可能是静态也可能是动态的,根据分派依据的宗量数可分为单分派和多分派。这两种分派方式的两两组合就构成了静态单分派,静态多分派,动态单分派,动态多分派这4种组合。本章讲静态分派。1、静态分派所有依赖静态类型来定位方法执行...
    99+
    2023-05-31
    jvm 方法调用 静态
  • java访问者模式的静态动态及伪动态分派实例分析
    今天小编给大家分享一下java访问者模式的静态动态及伪动态分派实例分析的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1 使用...
    99+
    2023-07-02
  • java访问者模式的静态动态及伪动态分派彻底理解
    目录1 使用访问者模式实现KPI考核的场景2 从静态分派到动态分派2.1 静态分派2.2 动态分派3 访问者模式中的伪动态分派4 访问者模式在JDK源码中的应用5 访问者模式在Spr...
    99+
    2024-04-02
  • java静态和非静态方法怎么调用
    在Java中,静态方法和非静态方法的调用方式有所不同。 静态方法调用: 静态方法可以直接通过类名来调用,不需要创建类的实例。例如:...
    99+
    2024-04-02
  • 使用JVM 如何实现动态分派功能
    这期内容当中小编将会给大家带来有关使用JVM 如何实现动态分派功能,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。1. 动态分派一个体现是重写(override)。下面的代码,运行结果很明显。public ...
    99+
    2023-05-31
    jvm 动态分派
  • 怎么使用java JVM方法分派模型
    这篇文章主要介绍了怎么使用java JVM方法分派模型的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇怎么使用java JVM方法分派模型文章都会有所收获,下面我们一起来看看吧。1. 知识储备...
    99+
    2023-07-02
  • kotlin怎么调用java静态方法
    在 Kotlin 中调用 Java 的静态方法非常简单,只需要使用 Java 类的全限定名加上静态方法名即可。例如,假设有一个 Ja...
    99+
    2024-04-02
  • Java中的静态字段和静态方法怎么用
    这篇文章主要介绍了Java中的静态字段和静态方法怎么用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。先看看下面这一串代码:public class Mai...
    99+
    2023-06-29
  • 静态分配ip和动态分配ip的方法是什么
    本篇内容主要讲解“静态分配ip和动态分配ip的方法是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“静态分配ip和动态分配ip的方法是什么”吧!在TCP/IP网络中,为保证计算机能够正常工作,...
    99+
    2023-06-21
  • C++ 虚拟函数与动态调派:理解运行时方法调用的机制
    虚拟函数允许派生类重写基类方法,动态调派则根据对象类型在运行时确定调用哪个函数。具体步骤包括:通过 virtual 声明虚拟函数,允许派生类重写。在派生类中重写虚拟函数,提供特定实现。使...
    99+
    2024-04-28
    c++ 虚拟函数
  • c#怎么调用静态方法
    在 c# 中,调用静态方法可以通过两种方式:1. 使用类名直接调用;2. 使用类型名称 + 双冒号调用。 如何在 C# 中调用静态方法 什么是静态方法? 静态方法是不属于任何特定对象实...
    99+
    2024-05-12
    c#
  • c#静态方法怎么调用
    非常抱歉,由于您没有提供文章标题,我无法为您生成一篇高质量的文章。请您提供文章标题,我将尽快为您生成一篇优质的文章。...
    99+
    2024-05-14
  • 如何使用Mockito调用静态方法和void方法
    一、mock 静态方法 mockito库并不能mock静态方法,需要依赖powermock 第一步:给类添加注解 // 静态类优先加载,所以需要提前告诉powermock哪些静态...
    99+
    2024-04-02
  • 怎么在java中使用静态方法
    本篇文章为大家展示了怎么在java中使用静态方法,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Java有哪些集合类Java中的集合主要分为四类:1、List列表:有序的,可重复的;2、Queue队列...
    99+
    2023-06-14
  • 使用Java怎么实现静态代理和动态代理
    本篇文章为大家展示了使用Java怎么实现静态代理和动态代理,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。代理模式代理模式(Proxy):为其他对象提供一个代理以控制对这个对象的访问。主要解决:在直接...
    99+
    2023-06-14
  • 一文分析php静态方法的优缺点和用法
    PHP 是一门灵活多变的编程语言,其内置的面向对象特性被广泛应用于 Web 开发中。其中静态方法是其中一个重要的特性。静态方法在 PHP 中有着很多的优势和劣势,本文将会分析静态方法的优点和缺点以及如何正确地使用它。一、静态方法的优点静态方...
    99+
    2023-05-14
  • php中的静态属性和静态方法怎么用
    这篇文章主要介绍“php中的静态属性和静态方法怎么用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“php中的静态属性和静态方法怎么用”文章能帮助大家解决问题。在php中,通过static关键字修饰的...
    99+
    2023-06-26
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作