百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Java字节码指令:invokeinterface(0xB9)

yuyutoo 2024-11-12 14:58 21 浏览 0 评论

invokeinterface 是 Java 字节码中的一条指令,用于调用接口方法。它在调用时根据对象的实际类型查找对应的方法实现,并执行它。invokeinterface 是为了解决多态性而引入的,因为接口的方法在编译时无法确定具体的实现类,因此需要在运行时进行查找。

invokeinterface(操作码) + n(常量池索引)+argc(参数个数)

  • 操作码:0xB9
  • 常量池索引:一个无符号的 16 位整数(范围 0 到 65,535 ),表示一个常量池索引。
  • 参数个数:一个无符号的 8 位整数(范围 0 到 255 ),表示调用接口方法时传递的参数个数。这个参数个数是指方法的所有参数,包括显式声明的参数和隐式的对象引用(即this引用,指向调用接口方法的对象实例),计算示例:如果接口方法签名是void myMethod(int x),那么invokeinterface指令的参数个数将是2(一个是this引用,一个是int参数 x )。
  • 功能:用于调用接口方法。
  • 操作数栈:需要确保操作数栈必需保证栈内存在对象引用的值,并且包含与参数个数相匹配个数的参数值(包括零个、一个或者多个参数值),且对象引用的值先入栈然后参数值入栈,否则指令无法正常执行。
  • 工作原理:
  1. 解析操作数:使用操作数中的常量池索引,从常量池中找到对应的 CONSTANT_InterfaceMethodref 项,CONSTANT_InterfaceMethodref 项包含了接口名称、方法名称和方法描述符。
  2. 弹出操作数栈中的数据:根据操作数中的参数个数字节从操作数栈中弹出方法所需数量的参数包括对象引用this。
  3. 执行指令首先,根据操作数栈中弹出的对象引用(即this)推导出对象运行时的实际对类型,在实际对象的类中查找与目标方法名称和描述符匹配的方法,如果该类没有实现该方法,则继续向上查找父类,直到找到匹配的方法。然后将对象引用和参数传递给找到的方法,并执行该方法的字节码指令。最后根据方法描述符提供的返回类型信息判断方法是否需要返回值,如果方法有返回值,则将返回值压入操作数栈,等待后续指令处理。
  • 示例:
invokeinterface #4, 1  //这条指令将用于调用常量池中索引为 #4 位置上关联的接口方法, 1 表示该方法参数的数量为 1 个(即this)。

使用示例

假设我们有以下Java代码:

public void exampleMethod() {
        ArrayList arrayList=new ArrayList<String>();
        arrayList.add("添加arrayList");
        List list=arrayList;
        list.add("添加list");
    }

编译成字节码后,相关部分可能如下:

0:  new             #2    // 创建一个新的 ArrayList 实例
3:  dup                   // 复制栈顶的引用(用于构造方法)
4:  invokespecial   #3    // 调用 ArrayList 的构造方法
7:  astore_1              // 将 ArrayList 对象存储到局部变量 1(arrayList)
8:  aload_1               // 将局部变量 1(arrayList)加载到栈顶
9:  ldc             #4    // 将字符串常量 "添加arrayList" 加载到栈顶
12: invokevirtual   #5    // 调用 ArrayList 的 add 方法
15: pop                   // 弹出栈顶的 32 位数据 (即布尔值)
16: aload_1               // 将局部变量 1(arrayList)加载到栈顶
17: astore_2              // 将 ArrayList 对象存储到局部变量 2(list)
18: aload_2               // 将局部变量 2(list)加载到栈顶
19: ldc             #6    // 将字符串常量 "添加list" 加载到栈顶
22: invokeinterface #7,2  // 调用 List 的 add 方法
27: pop                   // 弹出栈顶的 32 位数据 (即布尔值)
26: return                // 方法结束,返回

解释

  1. new #2:创建一个新的 ArrayList 对象。#2 是指向 ArrayList 类的常量池索引。
  2. dup:复制栈顶的对象引用,以便用于构造方法调用。
  3. invokespecial #3:调用 ArrayList 的构造方法来初始化新对象。#3 是构造方法的常量池索引。
  4. astore_1:将新创建的 ArrayList 对象存储到局部变量表的索引 1 中(即 arrayList)。
  5. aload_1:将局部变量 1(arrayList)的引用加载到操作数栈顶。
  6. ldc #4:将字符串常量 "添加arrayList" 加载到操作数栈顶。#4 是常量池中对应字符串的索引。
  7. invokevirtual #5:调用 ArrayList 的 add 方法(对象的非静态方法),将 "添加arrayList" 添加到列表中。#5 是 add 方法的常量池索引。
  8. pop:弹出栈顶的 32 位数据 (即布尔值),即之前调用对象的非静态方法,该方法压入操作数栈中的方法返回值。
  9. aload_1:将局部变量 1(arrayList)加载到操作数栈顶。
  10. astore_2:将操作数栈中的引用存储到局部变量表的索引 2 中(即 list)。
  11. aload_2:将局部变量 2(list)的引用加载到操作数栈顶。
  12. ldc #6:将字符串常量 "添加list" 加载到操作数栈顶。#6 是常量池中对应字符串的索引。
  13. invokeinterface #7,2:调用 List 的 add 方法(接口方法),将 "添加list" 添加到列表中。#7 是 add 方法的常量池索引。2 是传递的参数个数(即 "添加list"和对象引用this)。
  14. pop:弹出栈顶的 32 位数据 (即布尔值),即之前调用对象的非静态方法,该方法压入操作数栈中的方法返回值。
  15. return:结束方法的执行并返回(由于方法的返回类型为 void,不需要返回值)。

相关推荐

对volatile,synchronized,AQS的加锁解锁原理的一些理解

一、为什么要加锁,要实现同步多线程编程中,有可能会出现多个线程同时访问同一个共享、可变资源的情况,这个资源我们称之其为临界资源;这种资源可能是:对象、变量、文件等。...

注意,不能错过的CAS+volatile实现同步代码块

前言:最近看到有人说可以使用CAS+volatile实现同步代码块。心想,确实是可以实现的呀!因为AbstractQueuedSynchronizer(简称AQS)内部就是通过CAS+...

面试并发volatile关键字时,我们应该具备哪些谈资?

提前发现更多精彩内容,请访问https://dayarch.top/提前发现更多精彩内容,请访问https://dayarch.top/提前发现更多精彩内容,请访问https://dayarch...

无锁同步-JAVA之Volatile、Atomic和CAS

1、概要本文是无锁同步系列文章的第二篇,主要探讨JAVA中的原子操作,以及如何进行无锁同步。关于JAVA中的原子操作,我们很容易想到的是Volatile变量、java.util.concurrent....

C/C++面试题(二):std::atomic与volatile

volatile是C/C++中的一个关键字,用于告知编译器某个变量的值可能会在程序的控制之外被意外修改(例如被硬件、中断服务程序、多线程环境或其他外部代理)。为了防止编译器对代码进行某些可能破坏...

VOCs(Volatile Organic Compounds)挥发性有机化合物及测试方法

经常看到一些三防漆、涂料、油漆类产品的介绍中提到VOC、VOCs等概念,那么什么是VOC、VOCs和TVOC,VOCs主要包括哪些物质?VOCs的来源有哪些?VOCs的危害及国家标准是什么?一、V...

对volatile 及happen—before的理解

happen—before规则介绍Java...

这一篇我们来了解Synchronized、Volatile、Final关键字

题外话:蓝银王觉醒了!!--来自于一个斗罗大陆动漫爱好者(鹅,打钱!)湿兄这两天回家了,办了点大事,回来的时候我弟弟还舍不得我,哭着不愿意让我回京(我弟还是小学),我也心里很不舍,但是还是要回京奋斗...

关于 Java 关键字 volatile 的总结

1什么是volatilevolatile是Java的一个关键字,它提供了一种轻量级的同步机制。相比于重量级锁synchronized,volatile更为轻量级,因为它不会引起线程上下文...

大白话聊聊Java并发面试问题之volatile到底是什么?

用最简单的大白话,加上多张图给大家说一下,volatile到底是什么?...

为什么要有volatile关键字(volatile 关键字为什么不能保证原子性)

在嵌入式编程和多线程编程中,我们常会见到volatile关键字声明的变量。下面说一下volatile关键字的作用:1.保持变量内存可见简而言之就是用volatile声明的变量会告诉编译器和处理器,这个...

Java的volatile到底怎么理解?(java volatitle)

我们都知道,在Java中有很多的关键字,比如synchronize比如volatile,这些都是一些比较关键的,还有final,今天我们就来聊一下这个volatile因为这个vo...

Java多线程编程中的volatile关键字:解密神秘的共享内存

Java多线程编程中的volatile关键字:解密神秘的共享内存在Java多线程编程的世界里,volatile关键字就像一位低调却至关重要的守护者。它默默无闻地站岗放哨,确保多个线程之间能够正确地共享...

你了解volatile关键字的作用吗?(关键字volatile有什么含意?并举出三个不同的例子?)

【死记硬背】volatile关键字主要用于保持内存的变量可见性和禁止重排序。变量可见性:当一个线程改变了变量的值,那么新的值对于其他线程也是可以立即获取到的。禁止重排序:...

谈谈你对volatile 关键字作用和原理的理解

一位6年工作经验的小伙伴,在某里二面的时候被问到“volatile”关键字。然后,就没有然后了…同样,还有一位4年的小伙伴,去某团面试也被问到“volatile关键字“。然后,也没有然后了…...

取消回复欢迎 发表评论: