《深入Java虚拟机》笔记:指令集 (转)

《深入Java虚拟机》笔记:指令集

指令 含义
iconst_m1 把int型常量-1压入栈中
iconst_0 把int型常量压入栈中
fconst_1 把float型常量1压入栈中
lconst_2 把long型常量2压入栈中
dconst_3 把double型常量3压入栈中
bipush byte1 把byte1转换成int型压入栈中
sipush byte1,byte2 把byte1,byte2组成的short转换成int压入栈中
aconst_null 把空对象压入栈中
ldc indexbyte1 把常量池中指定下标的一个字长压入栈
ldc_w indexbyte1,indexbyte2 把常量池中指定下标的一个字长压入栈
ldc2_w indexbyte1,intexbyte2 把常量池中指定下标的两个字长压入栈
pop 弹出栈顶的一个字
pop2 弹出栈顶的两个字
swap 交换栈顶的两个字
dup 复制栈顶的一个字
dup2 复制栈顶的两个字
dup_x1 复制一字,并将弹出的两字压栈
dup_x2 复制一字,并将弹出的三字压栈
dup2_x1 复制两字,并将弹出的三字压栈
dup2_x2 复制两字,并将弹出的四字压栈
iload vindex 将位置vindex的int型局部变量压入栈
iload_0 将位置0的int型局部变量压入栈
fload vindex 将位置vindex的float型局部变量压入栈
fload_0 将位置0的float型局部变量压入栈
lload vindex 将位置vindex的long型局部变量压入栈
lload_0 将位置0的long型局部变量压入栈
dload vindex 将位置vindex的double型局部变量压入栈
dload_0 将位置0的double型局部变量压入栈
aload vindex 将位置vindex的对象引用局部变量压入栈
aload_0 将位置0的对象引用局部变量压入栈
istore vindex 弹出int型,存储在vindex处的局部变量
istore_0 弹出int型,存储在0处的局部变量
fstore vindex 弹出float型,存储在vindex处的局部变量
fstore_0 弹出float型,存储在0处的局部变量
lstore vindex 弹出long型,存储在vindex处的局部变量
lstore_0 弹出long型,存储在0处的局部变量
dstore vindex 弹出double型,存储在vindex处的局部变量
dstore_0 弹出double型,存储在0处的局部变量
astore vindex 弹出对象引用类型,存储在vindex处的局部变量
astore_0 弹出对象引用类型,存储在0处的局部变量
wide iload byte1,byte2 wide能把原来的一字节的参数变成处理两字节
i2l 把int类型转换成long类型,其他类似,不再举例
iadd 弹出两个int型,相加后结果压入栈
ladd 弹出两个long型,相加后结果压入栈
iinc vindex,const 把常量const与位于vindex的局部变量相加
wide iinc index1,intdex2,const1,const2 把常量与位于vindex1,vindex2的局部变量相加
isub 弹出两个int型,相减后lint型结果压入栈
lsub 弹出两个long型,相减后long型结果压入栈
imul 弹出两个int型,相乘后int型结果压入栈
lmul 弹出两个long型,相乘后long型结果压入栈
idev 弹出两个int型,相除后int型结果压入栈
ldev 弹出两个long型,相除后long型结果压入栈
irem 弹出两个int型,取余后int型结果压入栈
lrem 弹出两个long型,取余后long型结果压入栈
ineg 弹出一个int型,取反后int型结果压入栈
lneg 弹出一个long型,取反后long型结果压入栈
ishl 向左对int型值进行移位操作(long类似)
ishr 向右对int型值进行算术移位操作
iushr 向右对int型值进行逻辑移位操作
iand 对两个int类型值进行逻辑与运算(long类似)
ior 对两个int类型值进行逻辑或运算
ixor 对两个int类型值进行逻辑异或运算
fadd 弹出两个float型,相加后float型结果压入栈
fsub 弹出两个float型,相减后float型结果压入栈
fmul 弹出两个float型,相乘后float型结果压入栈
fdiv 弹出两个float型,相除后float型结果压入栈
frem 弹出两个float型,取余后float型结果压入栈
new indexbyte1,indexbyte2 在堆中创建一个新的对象,将其压入栈中
putfield indexbyte1,intdebyte2 设置对象字段的值,值和对象引用都从栈获取
getfield indexbyte1,indexbyte2 将对象字段压入栈,对象引用从栈中获取
putstatic indexbyte1,indexbyte2 设置类静态变量的值,从栈中获取
getstatic indexbyte1,indexbyte2 将静态变量压入栈
checkcast indexbyte1,indexbyte2 如果栈中的引用不是指定位置的类,则抛出异常
instanceof indexbyte1,indexbyte2 同上,不过是往结果中压入true或false
newarray atype 从栈中弹出长度,分配atype类型的数组
anewarray indexbyte1,intdexbyte2 从栈中取长度,index取类型
multianewarray index1,index2,dimensions 分配多维数组
arraylength 从栈中弹出一个引用,并将其长度压入栈
baload 从栈中取数组引用和下标,把对应boolean压入栈
caload 同上,char
saload 同上,short
iaload 同上,int
laload 同上,long
faload 同上,folat
daload 同上,double
aaload 同上,array
bastore
将byte或boolean类型的值value、索引index、数组引用arrayref弹出栈,赋值为arrayref[index]=value

其他类型同上

ifeq branchbyte1,branchbyte2 从栈中弹出int型,为0则跳转到指定的分支执行
ifne branchbyte1,branchbyte2 同上,不为0则跳转到指定的分支执行
iflt branchbyte1,branchbyte2 同上,如果小于0,则跳转到指定的分支执行
ifle branchbyte1,branchbyte2 同上,如果小于等于0,则跳转到指定的分支执行
ifgt branchbyte1,branchbyte2 同上,如果大于0,则跳转到指定的分支执行
ifge branchbyte1,branchbyte2 同上,如果大于等于0,则跳转到指定的分支执行
if_icmpeq branchbyte1,branchbyte2 栈中弹出两个int,如果的一个等于第二个,跳转
if_icmpme branchbyte1,branchbyte2 栈中弹出两个int,如果的一个不等于第二个,跳转
if_icmplt branchbyte1,branchbyte2 栈中弹出两个int,如果的一个小于第二个,跳转
if_icmple branchbyte1,branchbyte2 栈中弹出两个int,如果的一个小于等于第二个,跳转
if_icmpgt branchbyte1,branchbyte2 栈中弹出两个int,如果的一个大于第二个,跳转
if_icmpge branchbyte1,branchbyte2 栈中弹出两个int,如果的一个大于等于第二个,跳转
lcmp 从栈中弹出两个long,把结果-1、0、1放入栈
fcmpg 从栈中弹出两个float,把结果-1、0、1放入栈
fcmpl 从栈中弹出两个float,把结果-1、0、1放入栈
dcmpg 从栈中弹出两个double,把结果-1、0、1放入栈
dcmpl 从栈中弹出两个double,把结果-1、0、1放入栈
ifnull branchbyte1,branchbyte2 从栈中弹出,如果是null就跳到分支
ifnonnull branchbyte1,branchbyte2 从栈中弹出,如果不是null就跳到分支
时间: 2024-08-06 04:59:47

《深入Java虚拟机》笔记:指令集 (转)的相关文章

Java虚拟机笔记 – JVM 自定义的类加载器的实现和使用2

1.用户自定义的类加载器: 要创建用户自己的类加载器,只需要扩展java.lang.ClassLoader类,然后覆盖它的findClass(String name)方法即可,该方法根据参数指定类的名字,返回对应的Class对象的引用. findClass protected Class<?> findClass(String name) throws ClassNotFoundException 使用指定的二进制名称查找类.此方法应该被类加载器的实现重写,该实现按照委托模型来加载类.在通过父

Java虚拟机笔记 – JVM 自定义的类加载器的实现和使用

1.用户自定义的类加载器: 要创建用户自己的类加载器,只需要扩展java.lang.ClassLoader类,然后覆盖它的findClass(String name)方法即可,该方法根据参数指定类的名字,返回对应的Class对象的引用. findClass protected Class<?> findClass(String name) throws ClassNotFoundException 使用指定的二进制名称查找类.此方法应该被类加载器的实现重写,该实现按照委托模型来加载类.在通过父

深入理解java虚拟机笔记(一)-java内存区域与内存溢出

1. 前言 这是深入理解java虚拟机一书的笔记,来自第二章.因为这本书讲的比较深奥,这是第二次看,需要记录一下笔记. 2. 运行时数据区域 java虚拟机所管理的内存分为以下几个区域. ps:图片来自网络 2.1 程序计数器 程序计数器是一块较小的内存空间,他可以看做是当前线程所执行字节码的行号指示器.字节码解释器工作时就是通过改变这个计数器的值来选去下一条要执行的字节码指令,分之.循环.跳转.异常处理.线程恢复等基础功能都需要依赖这个计数器来完成. 这块内存是线程私有的内存. 如果线程在执行

Java虚拟机笔记(二):GC垃圾回收

为什么要了解GC 我们都知道Java开发者在开发过程中是不需要关心对象的回收的,因为Java虚拟机的原因,它会自动回收那些失效的垃圾对象.那我们为什么还要去了解GC和内存分配呢? 答案很简单:当我们需要排查各种内存溢出.内存泄漏时,当垃圾收集器成为系统达到更高并发量的瓶颈时,我们就需要对这些"自动化"的技术实施必要的监控和调节. 回收哪些对象 我们知道在Java内存运行时数据区域中,虚拟机栈.本地方法栈和程序计数器是线程隔离的数据区,随线程而生,随线程而灭:栈中的栈帧随着方法的进入和退

深入理解Java虚拟机--笔记1

Java内存区域与内存溢出异常 运行时数据区域 Java虚拟机在执行Java程序的过程中会把它所管理的内存区域划分为若干个不同的数据区域. 1 程序计数器--Program Counter Register 一块较小的内存空间,作用可视为当前线程所执行的字节码的行号指示器,没个线程有自己独立的程序计数器,即"线程私有"的内存区域. 若线程当前执行的是一个Java方法,则这个计数器记录的是当前执行的虚拟机字节码指令的地址: 若线程当前执行的是一个Native方法,则这个计数器值为空(Un

Java虚拟机笔记(一):类加载机制

一.概述 虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 二.类加载的生命周期 类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading).验证(Verification).准备(Preparation).解析(Resolution).初始化(Initialization).使用(Using)和卸载(Unloading)7个阶段. 其中:验证.准备.解析3

深入理解Java虚拟机笔记---属性表集合

在Class文件,字段表,方法表中都可以携带自己的属性表集合,以用于描述某些场景专有的信息.与Class文件中其它的数据项目要求的顺序.长度和内容不同,属性表集合的限制稍微宽松一些,不再要求各个属性表具有严格的顺序,并且只要不与已有的属性名重复,任何人实现的编译器都可以向属性表中写入自己定义的属性信息,Java虚拟机运行时会忽略掉它不认识的属性.为了能正确地解析Class文件,<Java虚拟机规范(第二版)>中预定义了9荐虚拟机实现应当能识别的属性,具体如下表所示: 对于每个属性,它的名称需要

深入理解Java虚拟机笔记---方法调用

方法调用并不等同于方法执行,方法调用阶段唯一的任务就是确定调用方法的版本(即调用哪一个方法),暂时还不涉及方法内部的具体运行过程.在程序运行时,进行方法调用是最普遍.最频繁的操作.在Class文件的编译过程中不包含传统编译中的连接步骤,一切方法调用在Class文件里存储的都只是符号引用,而不是方法在实际运行时内存布局中的入口地址(相当于直接引用).这个特性给Java带来了更强大的动态扩展能力,但也使得Java方法的调用过程变得相对复杂,需要在类加载期间甚至到运行期间才能确定目标方法的直接引用.

深入理解Java虚拟机笔记---运行时栈帧结构

栈帧(Stack Frame)是用于支持虚拟机进行方法调用和方法执行的数据结构,它是虚拟机运行时数据区的虚拟机栈(Virtual Machine Stack)的栈元素.栈帧存储了方法的局部变量表,操作数栈,动态连接和方法返回地址等信息.第一个方法从调用开始到执行完成,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程. 每一个栈帧都包括了局部变量表,操作数栈,动态连接,方法返回地址和一些额外的附加信息.在编译代码的时候,栈帧中需要多大的局部变量表,多深的操作数栈都已经完全确定了,并且写入到了方法表的

深入理解Java虚拟机笔记---内存区域

Java虚拟机在执行Java程序过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域有自各的用途,以及创建及销毁时间,有的区域随着虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束而建立和销毁.根据<Java虚拟机规范(第2版)>规定,Java虚拟机管理的内存区域包括以下几个运行时数据区域,下如图 1.程序计数器(Program Counter Register) 程序计数器是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的行号指示器.字节码解释器工作时就是通过