JVM 字节码指令

JVM 字节码指令

Java源码在运行之前都要编译成为字节码格式(如.class文件),然后由ClassLoader将字节码载入运行。在字节码文件中,指令代码只是其中的一部分,里面还记录了字节码文件的编译版本、常量池、访问权限、所有成员变量和成员方法等信息(详见Java字节码规范)。本文主要简单介绍不同Java指令的功能。

虚拟机栈操作

常量入栈指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x01aconst_nullnull值入栈。
0x02iconst_m1-1(int)值入栈。
0x03iconst_00(int)值入栈。
0x04iconst_11(int)值入栈。
0x05iconst_22(int)值入栈。
0x06iconst_33(int)值入栈。
0x07iconst_44(int)值入栈。
0x08iconst_55(int)值入栈。
0x09lconst_00(long)值入栈。
0x0alconst_11(long)值入栈。
0x0bfconst_00(float)值入栈。
0x0cfconst_11(float)值入栈。
0x0dfconst_22(float)值入栈。
0x0edconst_00(double)值入栈。
0x0fdconst_11(double)值入栈。
0x10bipushvaluebytevaluebyte值带符号扩展成int值入栈。
0x11sipushvaluebyte1(valuebyte1 << 8) | valuebyte2 值带符号扩展成int值入栈。
valuebyte2
0x12ldcindexbyte1常量池中的常量值(int, float, string reference, object reference)入栈。
0x13ldc_windexbyte1常量池中常量(int, float, string reference, object reference)入栈。
indexbyte2
0x14ldc2_windexbyte1常量池中常量(long, double)入栈。
indexbyte2

局部变量入栈

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x19(wide)aloadindexbyte从局部变量indexbyte中装载引用类型值入栈。
0x2aaload_0从局部变量0中装载引用类型值入栈。
0x2baload_1从局部变量1中装载引用类型值入栈。
0x2caload_2从局部变量2中装载引用类型值入栈。
0x2daload_3从局部变量3中装载引用类型值入栈。
0x15(wide)iloadindexbyte从局部变量indexbyte中装载int类型值入栈。
0x1aiload_0从局部变量0中装载int类型值入栈。
0x1biload_1从局部变量1中装载int类型值入栈。
0x1ciload_2从局部变量2中装载int类型值入栈。
0x1diload_3从局部变量3中装载int类型值入栈。
0x16(wide)lloadindexbyte从局部变量indexbyte中装载long类型值入栈。
0x1elload_0从局部变量0中装载int类型值入栈。
0x1flload_1从局部变量1中装载int类型值入栈。
0x20lload_2从局部变量2中装载int类型值入栈。
0x21lload_3从局部变量3中装载int类型值入栈。
0x17(wide)floadindexbyte从局部变量indexbyte中装载float类型值入栈。
0x22fload_0从局部变量0中装载float类型值入栈。
0x23fload_1从局部变量1中装载float类型值入栈。
0x24fload_2从局部变量2中装载float类型值入栈。
0x25fload_3从局部变量3中装载float类型值入栈。
0x18(wide)dloadindexbyte从局部变量indexbyte中装载double类型值入栈。
0x26dload_0从局部变量0中装载double类型值入栈。
0x27dload_1从局部变量1中装载double类型值入栈。
0x28dload_2从局部变量2中装载double类型值入栈。
0x29dload_3从局部变量3中装载double类型值入栈。
0x32aaload从引用类型数组中装载指定项的值。
0x2eiaload从int类型数组中装载指定项的值。
0x2flaload从long类型数组中装载指定项的值。
0x30faload从float类型数组中装载指定项的值。
0x31daload从double类型数组中装载指定项的值。
0x33baload从boolean类型数组或byte类型数组中装载指定项的值(先转换为int类型值,后压栈)。
0x34caload从char类型数组中装载指定项的值(先转换为int类型值,后压栈)。
0x35saload从short类型数组中装载指定项的值(先转换为int类型值,后压栈)。

栈顶值保存到局部变量

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x3a(wide)astoreindexbyte将栈顶引用类型值保存到局部变量indexbyte中。
0x4bastroe_0将栈顶引用类型值保存到局部变量0中。
0x4castore_1将栈顶引用类型值保存到局部变量1中。
0x4dastore_2将栈顶引用类型值保存到局部变量2中。
0x4eastore_3将栈顶引用类型值保存到局部变量3中。
0x36(wide)istoreindexbyte将栈顶int类型值保存到局部变量indexbyte中。
0x3bistore_0将栈顶int类型值保存到局部变量0中。
0x3cistore_1将栈顶int类型值保存到局部变量1中。
0x3distore_2将栈顶int类型值保存到局部变量2中。
0x3eistore_3将栈顶int类型值保存到局部变量3中。
0x37(wide)lstoreindexbyte将栈顶long类型值保存到局部变量indexbyte中。
0x3flstore_0将栈顶long类型值保存到局部变量0中。
0x40lstore_1将栈顶long类型值保存到局部变量1中。
0x41lstore_2将栈顶long类型值保存到局部变量2中。
0x42lstroe_3将栈顶long类型值保存到局部变量3中。
0x38(wide)fstoreindexbyte将栈顶float类型值保存到局部变量indexbyte中。
0x43fstore_0将栈顶float类型值保存到局部变量0中。
0x44fstore_1将栈顶float类型值保存到局部变量1中。
0x45fstore_2将栈顶float类型值保存到局部变量2中。
0x46fstore_3将栈顶float类型值保存到局部变量3中。
0x39(wide)dstoreindexbyte将栈顶double类型值保存到局部变量indexbyte中。
0x47dstore_0将栈顶double类型值保存到局部变量0中。
0x48dstore_1将栈顶double类型值保存到局部变量1中。
0x49dstore_2将栈顶double类型值保存到局部变量2中。
0x4adstore_3将栈顶double类型值保存到局部变量3中。
0x53aastore将栈顶引用类型值保存到指定引用类型数组的指定项。
0x4fiastore将栈顶int类型值保存到指定int类型数组的指定项。
0x50lastore将栈顶long类型值保存到指定long类型数组的指定项。
0x51fastore将栈顶float类型值保存到指定float类型数组的指定项。
0x52dastore将栈顶double类型值保存到指定double类型数组的指定项。
0x54bastroe将栈顶boolean类型值或byte类型值保存到指定boolean类型数组或byte类型数组的指定项。
0x55castore将栈顶char类型值保存到指定char类型数组的指定项。
0x56sastore将栈顶short类型值保存到指定short类型数组的指定项。

wide指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xc4wide使用附加字节扩展局部变量索引(iinc指令特殊)。

通用(无类型)栈操作指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x00nop空操作。
0x57pop从栈顶弹出一个字长的数据。
0x58pop2从栈顶弹出两个字长的数据。
0x59dup复制栈顶一个字长的数据,将复制后的数据压栈。
0x5adup_x1复制栈顶一个字长的数据,弹出栈顶两个字长数据,先将复制后的数据压栈,再将弹出的两个字长数据压栈。
0x5bdup_x2复制栈顶一个字长的数据,弹出栈顶三个字长的数据,将复制后的数据压栈,再将弹出的三个字长的数据压栈。
0x5cdup2复制栈顶两个字长的数据,将复制后的两个字长的数据压栈。
0x5ddup2_x1复制栈顶两个字长的数据,弹出栈顶三个字长的数据,将复制后的两个字长的数据压栈,再将弹出的三个字长的数据压栈。
0x5edup2_x2复制栈顶两个字长的数据,弹出栈顶四个字长的数据,将复制后的两个字长的数据压栈,再将弹出的四个字长的数据压栈。
0x5fswap交换栈顶两个字长的数据的位置。Java指令中没有提供以两个字长为单位的交换指令。

类型转换指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x86i2f将栈顶int类型值转换为float类型值。
0x85i2l将栈顶int类型值转换为long类型值。
0x87i2d将栈顶int类型值转换为double类型值。
0x8bf2i将栈顶float类型值转换为int类型值。
0x8cf2l将栈顶float类型值转换为long类型值。
0x8df2d将栈顶float类型值转换为double类型值。
0x88l2i将栈顶long类型值转换为int类型值。
0x89l2f将栈顶long类型值转换为float类型值。
0x8al2d将栈顶long类型值转换double类型值。
0x8ed2i将栈顶double类型值转换为int类型值。
0x90d2f将栈顶double类型值转换为float类型值。
0x8fd2l将栈顶double类型值转换为long类型值。
0x91i2b将栈顶int类型值截断成byte类型,后带符号扩展成int类型值入栈。
0x92i2c将栈顶int类型值截断成char类型值,后带符号扩展成int类型值入栈。
0x93i2s将栈顶int类型值截断成short类型值,后带符号扩展成int类型值入栈。

数值运算

整数运算

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x60iadd将栈顶两int类型数相加,结果入栈。
0x64isub将栈顶两int类型数相减,结果入栈。
0x68imul将栈顶两int类型数相乘,结果入栈。
0x6cidiv将栈顶两int类型数相除,结果入栈。
0x70irem将栈顶两int类型数取模,结果入栈。
0x74ineg将栈顶int类型值取负,结果入栈。
0x61ladd将栈顶两long类型数相加,结果入栈。
0x65lsub将栈顶两long类型数相减,结果入栈。
0x69lmul将栈顶两long类型数相乘,结果入栈。
0x6dldiv将栈顶两long类型数相除,结果入栈。
0x71lrem将栈顶两long类型数取模,结果入栈。
0x75lneg将栈顶long类型值取负,结果入栈。
0x84(wide)iincindexbyte将整数值constbyte加到indexbyte指定的int类型的局部变量中。
constbyte

浮点运算

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x62fadd将栈顶两float类型数相加,结果入栈。
0x66fsub将栈顶两float类型数相减,结果入栈。
0x6afmul将栈顶两float类型数相乘,结果入栈。
0x6efdiv将栈顶两float类型数相除,结果入栈。
0x72frem将栈顶两float类型数取模,结果入栈。
0x76fneg将栈顶float类型值取反,结果入栈。
0x63dadd将栈顶两double类型数相加,结果入栈。
0x67dsub将栈顶两double类型数相减,结果入栈。
0x6bdmul将栈顶两double类型数相乘,结果入栈。
0x6fddiv将栈顶两double类型数相除,结果入栈。
0x73drem将栈顶两double类型数取模,结果入栈。
0x77dneg将栈顶double类型值取负,结果入栈。

逻辑运算

逻辑运算——移位运算

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x78ishl左移int类型值。
0x79lshl左移long类型值。
0x7aishr算术右移int类型值。
0x7blshr算术右移long类型值。
0x7ciushr逻辑右移int类型值。
0x7dlushr逻辑右移long类型值。

逻辑运算——按位布尔运算

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x73iand对int类型按位与运算。
0x7fland对long类型的按位与运算。
0x80ior对int类型的按位或运算。
0x81lor对long类型的按位或运算。
0x82ixor对int类型的按位异或运算。
0x83lxor对long类型的按位异或运算。

控制流指令

控制流指令——条件跳转指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x99ifeqbranchbyte1若栈顶int类型值为0则跳转。
branchbyte2
0x9aifnebranchbyte1若栈顶int类型值不为0则跳转。
branchbyte2
0x9bifltbranchbyte1若栈顶int类型值小于0则跳转。
branchbyte2
0x9eiflebranchbyte1若栈顶int类型值小于等于0则跳转。
branchbyte2
0x9difgtbranchbyte1若栈顶int类型值大于0则跳转。
branchbyte2
0x9cifgebranchbyte1若栈顶int类型值大于等于0则跳转。
branchbyte2
0x9fif_icmpeqbranchbyte1若栈顶两int类型值相等则跳转。
branchbyte2
0xa0if_icmpnebranchbyte1若栈顶两int类型值不相等则跳转。
branchbyte2
0xa1if_icmpltbranchbyte1若栈顶两int类型值前小于后则跳转。
branchbyte2
0xa4if_icmplebranchbyte1若栈顶两int类型值前小于等于后则跳转。
branchbyte2
0xa3if_icmpgtbranchbyte1若栈顶两int类型值前大于后则跳转。
branchbyte2
0xa2if_icmpgebranchbyte1若栈顶两int类型值前大于等于后则跳转。
branchbyte2
0xc6ifnullbranchbyte1若栈顶引用值为null则跳转。
branchbyte2
0xc7ifnonnullbranchbyte1若栈顶引用值不为null则跳转。
branchbyte2
0xa5if_acmpeqbranchbyte1若栈顶两引用类型值相等则跳转。
branchbyte2
0xa6if_acmpnebranchbyte1若栈顶两引用类型值不相等则跳转。
branchbyte2

控制流指令——比较指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0x94lcmp比较栈顶两long类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈。
0x95fcmpl比较栈顶两float类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈。
0x96fcmpg比较栈顶两float类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈。
0x97dcmpl比较栈顶两double类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈。
0x98dcmpg比较栈顶两double类型值,前者大,1入栈;相等,0入栈;后者大,-1入栈;有NaN存在,-1入栈。

控制流指令——无条件跳转指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xa7gotobranchbyte1无条件跳转到指定位置。
branchbyte2
0xc8goto_wbranchbyte1无条件跳转到指定位置(宽索引)。
branchbyte2
branchbyte3

控制流指令——表跳转指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xaatableswitch<0-3bytepad>通过索引访问跳转表,并跳转。
defaultbyte1
defaultbyte2
defaultbyte3
defaultbyte4
lowbyte1
lowbyte2
lowbyte3
lowbyte4
highbyte1
highbyte2
highbyte3
highbyte4
jump offsets...
0xablookupswitch<0-3bytepad>通过键值访问跳转表,并跳转。
defaultbyte1
defaultbyte2
defaultbyte3
defaultbyte4
npairs1
npairs2
npairs3
npairs4
match offsets

控制流指令——异常和finally

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xbfathrow抛出异常。
0xa8jsrbranchbyte1跳转到子例程序。
branchbyte2
0xc9jsr_wbranchbyte1跳转到子例程序(宽索引)。
branchbyte2
branchbyte3
branchbyte4
0xa9(wide)retindexbyte返回子例程序。

对象操作指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xbbnewindexbyte1创建新的对象实例。
indexbyte2
0xc0checkcastindexbyte1类型强转。
indexbyte
0xc1instanceofindexbyte1判断类型。
indexbyte2
0xb4getfieldindexbyte1获取对象字段的值。
indexbyte2
0xb5putfieldindexbyte1给对象字段赋值。
indexbyte2
0xb2getstaticindexbyte1获取静态字段的值。
indexbyte2
0xb3putstaticindexbyte1给静态字段赋值。
indexbyte2

数组操作指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xbcnewarrayatype创建type类型的数组。
0xbdanewarrayindexbyte1创建引用类型的数组。
indexbyte2
0xbearraylength获取一维数组的长度。
0xc5multianewarrayindexbyte1创建dimension维度的数组。
indexbyte2
dimension

方法调用指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xb7invokespecialindexbyte1编译时方法绑定调用方法。
indexbyte2
0xb6invokevirtualindexbyte1运行时方法绑定调用方法。
indexbyte2
0xb8invokestaticindexbyte1调用静态方法。
indexbyte2
0xb9invokeinterfaceindexbyte1调用接口方法。
indexbyte2
count
0

方法返回指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xacireturn返回int类型值。
0xadlreturn返回long类型值。
0xaefreturn返回float类型值。
0xafdreturn返回double类型值。
0xb0areturn返回引用类型值。
0xb1returnvoid函数返回。

线程同步指令

指令码操作码(助记符)操作数描述(栈指操作数栈)
0xc2monitorenter进入并获得对象监视器。
0xc3monitorexit释放并退出对象监视器。