【JVM原理探索】class字节码指令操作介绍(上)

简介: 【JVM原理探索】class字节码指令操作介绍(上)

前提概要


Java虚拟机的指令由一个字节长度、代表着某种特定操作含义的数字(称为操作码, Opcode)以及跟随其后的零至多个代表此操作所需参数(称为操作数,Operands)而构 成。




指令介绍


由于限制了Java虚拟机操作码的长度为一个字节(即 0~255),这意味着指令集的操作码总数不可能超过256条。


大多数的指令都包含了其操作所对应的数据类型信息,例如


  • iload 指令用于从局部变量表中加载int型的数据到操作数栈中
  • fload 指令加载的则是float 类型的数据。
  • 大部分的指令都没有支持整数类型 byte、char 和 short,甚至没有任何指令支持 boolean类型。
  • 大多数对于 boolean、byte、short 和 char 类型数据的操作,实际上都是使用相应的 int 类型作为运算类型



加载和存储指令


用于将数据在栈帧中的局部变量表和操作数栈之间来回传输,这类指令包括如下内容。



将一个局部变量加载到操作栈

iload、iload_<n>、
lload、lload_<n>、
fload、fload_<n>、
dload、dload_<n>、
aload、aload_<n>。
复制代码



将一个数值从操作数栈存储到局部变量表

istore、istore_<n>、
lstore、lstore_<n>、
fstore、fstore_<n>、
dstore、dstore_<n>、
astore、astore_<n>。
复制代码



将一个常量加载到操作数栈

bipush、sipush、
ldc、ldc_w、ldc2_w、
aconst_null、
iconst_m1、iconst_<i>、lconst_<l>、
fconst_<f>、dconst_<d>。
复制代码



扩充局部变量表的访问索引的指令

wide




运算或算术指令


用于对两个操作数栈上的值进行某种特定运算,并把结果重新存入到操作栈顶

加法指令:iadd、ladd、fadd、dadd。
减法指令:isub、lsub、fsub、dsub。
乘法指令:imul、lmul、fmul、dmul 等等
复制代码



类型转换指令


可以将两种不同的数值类型进行相互转换,Java 虚拟机直接支持以下数值类型的宽化类型转换 (即小范围类型向大范围类型的安全转换):


  • int 类型到 long、float 或者 double 类型。
  • long 类型到 float、double 类型。
  • float 类型到 double 类型。


处理窄化类型转换(Narrowing Numeric Conversions)时,必须显式地使用转换指令来完 成,这些转换指令包括:

i2b、i2c、i2s、l2i、f2i、f2l、d2i、d2l 和 d2f。
复制代码



创建类实例的指令:

new。


创建数组的指令:

newarray、anewarray、multianewarray。


访问字段指令:

getfield、putfield、getstatic、putstatic。


数组存取相关指令


把一个数组元素加载到操作数栈的指令:


baload、caload、
saload、iaload、
laload、faload、
daload、aaload。
复制代码



将一个操作数栈的值存储到数组元素中的指令:

bastore、castore、sastore、iastore、
fastore、dastore、aastore。
复制代码


取数组长度的指令:

arraylength。
复制代码


检查类实例类型的指令:

instanceof、checkcast。
复制代码



操作数栈管理指令


如同操作一个普通数据结构中的堆栈那样,Java 虚拟机提供了一些用于直接操作操作数栈 的指令,包括:


将操作数栈的栈顶一个或两个元素出栈

pop、pop2。
复制代码


复制栈顶一个或两个数值并将复制值或双份的复制值重新压入栈顶:

dup、dup2、dup_x1、dup2_x1、dup_x2、dup2_x2。
复制代码


将栈最顶端的两个数值互换:

swap
复制代码


控制转移指令


控制转移指令可以让 Java 虚拟机有条件或无条件地从指定的位置指令而不是控制转移指 令的下一条指令继续执行程序,从概念模型上理解,可以认为控制转移指令就是在有条件 或无条件地修改 PC 寄存器的值。控制转移指令如下。


条件分支:

ifeq、iflt、ifle、ifne、ifgt、ifge、ifnull、ifnonnull、if_icmpeq、if_icmpne、
if_icmplt、if_icmpgt、if_icmple、if_icmpge、if_acmpeq 和 if_acmpne。
复制代码


复合条件分支:

tableswitch、lookupswitch。
复制代码


无条件分支:

goto、goto_w、jsr、jsr_w、ret。
复制代码




方法调用指令


  • invokevirtual 指令用于调用对象的实例方法,根据对象的实际类型进行分派(虚方法分 派),这也是 Java 语言中最常见的方法分派方式。
  • invokeinterface 指令用于调用接口方法,它会在运行时搜索一个实现了这个接口方法的对 象,找出适合的方法进行调用。
  • invokespecial 指令用于调用一些需要特殊处理的实例方法,包括实例初始化方法、私有方法和父类方法。
  • invokestatic 指令用于调用类方法(static 方法)。
  • invokedynamic 指令用于在运行时动态解析出调用点限定符所引用的方法,并执行该方法,前面 4 条调用指令的分派逻辑都固化在 Java 虚拟机内部,而 invokedynamic 指令的分派逻辑是由用户所设定的引导方法决定的。


方法调用指令与数据类型无关。


方法返回指令


是根据返回值的类型区分的,包括

ireturn(当返回值是 boolean、byte、char、short 和 int
类型时使用)、
lreturn、freturn、dreturn 和 areturn,
另外还有一条 return 指令供声明为void 的方法、
实例初始化方法以及类和接口的类初始化方法使用。
复制代码



异常处理指令

在 Java 程序中显式抛出异常的操作(throw 语句)都由 athrow 指令来实现



Synchronized指令

有 monitorenter 和 monitorexit 两条指令来支持 synchronized 关键字的语义

image.png



相关文章
|
1月前
|
存储 监控 算法
美团面试:说说 G1垃圾回收 底层原理?说说你 JVM 调优的过程 ?
尼恩提示: G1垃圾回收 原理非常重要, 是面试的重点, 大家一定要好好掌握
美团面试:说说 G1垃圾回收 底层原理?说说你 JVM 调优的过程  ?
|
1月前
|
存储 SQL 小程序
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
这篇文章详细介绍了Java虚拟机(JVM)的运行时数据区域和JVM指令集,包括程序计数器、虚拟机栈、本地方法栈、直接内存、方法区和堆,以及栈帧的组成部分和执行流程。
33 2
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
|
1月前
|
SQL 缓存 Java
JVM知识体系学习三:class文件初始化过程、硬件层数据一致性(硬件层)、缓存行、指令乱序执行问题、如何保证不乱序(volatile等)
这篇文章详细介绍了JVM中类文件的初始化过程、硬件层面的数据一致性问题、缓存行和伪共享、指令乱序执行问题,以及如何通过`volatile`关键字和`synchronized`关键字来保证数据的有序性和可见性。
30 3
|
1月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
45 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
1月前
|
前端开发 Java 应用服务中间件
JVM进阶调优系列(1)类加载器原理一文讲透
本文详细介绍了JVM类加载机制。首先解释了类加载器的概念及其工作原理,接着阐述了四种类型的类加载器:启动类加载器、扩展类加载器、应用类加载器及用户自定义类加载器。文中重点讲解了双亲委派机制,包括其优点和缺点,并探讨了打破这一机制的方法。最后,通过Tomcat的实际应用示例,展示了如何通过自定义类加载器打破双亲委派机制,实现应用间的隔离。
|
3月前
|
Java
Java常见JVM虚拟机指令(47个)
Java常见JVM虚拟机指令(47个)
69 3
Java常见JVM虚拟机指令(47个)
|
3月前
|
缓存 前端开发 Java
浅析JVM invokedynamic指令与Java Lambda语法
【8月更文挑战第27天】在Java的演进历程中,invokedynamic指令的引入和Lambda表达式的出现无疑是两大重要里程碑。它们不仅深刻改变了Java的开发模式和性能表现,还极大地推动了Java在函数式编程和动态语言支持方面的进步。本文将从技术角度浅析JVM中的invokedynamic指令及其与Java Lambda语法的紧密联系。
58 0
|
4月前
|
Java Serverless 应用服务中间件
函数计算操作报错合集之JVM启动时找不到指定的日志目录,该如何解决
Serverless 应用引擎(SAE)是阿里云提供的Serverless PaaS平台,支持Spring Cloud、Dubbo、HSF等主流微服务框架,简化应用的部署、运维和弹性伸缩。在使用SAE过程中,可能会遇到各种操作报错。以下是一些常见的报错情况及其可能的原因和解决方法。
|
4月前
|
存储 监控 Java
揭秘Java虚拟机:探索JVM的工作原理与性能优化
本文深入探讨了Java虚拟机(JVM)的核心机制,从类加载到垃圾回收,再到即时编译技术,揭示了这些复杂过程如何共同作用于Java程序的性能表现。通过分析现代JVM的内存管理策略和性能监控工具,文章提供了实用的调优建议,帮助开发者有效提升Java应用的性能。
77 3
|
1月前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
37 4