JVM 指令集整理

指令码助记符说明
0×00nop什么都不做
0×01aconst_null将null推送至栈顶
0×02iconst_m1将int型-1推送至栈顶
0×03iconst_0将int型0推送至栈顶
0×04iconst_1将int型1推送至栈顶
0×05iconst_2将int型2推送至栈顶
0×06iconst_3将int型3推送至栈顶
0×07iconst_4将int型4推送至栈顶
0×08iconst_5将int型5推送至栈顶
0×09lconst_0将long型0推送至栈顶
0x0alconst_1将long型1推送至栈顶
0x0bfconst_0将float型0推送至栈顶
0x0cfconst_1将float型1推送至栈顶
0x0dfconst_2将float型2推送至栈顶
0x0edconst_0将double型0推送至栈顶
0x0fdconst_1将double型1推送至栈顶
0×10bipush将单字节的常量值(-128~127)推送至栈顶
0×11sipush将一个短整型常量值(-32768~32767)推送至栈顶
0×12ldc将int, float或String型常量值从常量池中推送至栈顶
0×13ldc_w将int, float或String型常量值从常量池中推送至栈顶(宽索引)
0×14ldc2_w将long或double型常量值从常量池中推送至栈顶(宽索引)
0×15iload将指定的int型本地变量推送至栈顶
0×16lload将指定的long型本地变量推送至栈顶
0×17fload将指定的float型本地变量推送至栈顶
0×18dload将指定的double型本地变量推送至栈顶
0×19aload将指定的引用类型本地变量推送至栈顶
0x1aiload_0将第0个int型本地变量推送至栈顶
0x1biload_1将第1个int型本地变量推送至栈顶
0x1ciload_2将第2个int型本地变量推送至栈顶
0x1diload_3将第3个int型本地变量推送至栈顶
0x1elload_0将第0个long型本地变量推送至栈顶
0x1flload_1将第1个long型本地变量推送至栈顶
0×20lload_2将第2个long型本地变量推送至栈顶
0×21lload_3将第3个long型本地变量推送至栈顶
0×22fload_0将第0个float型本地变量推送至栈顶
0×23fload_1将第1个float型本地变量推送至栈顶
0×24fload_2将第2个float型本地变量推送至栈顶
0×25fload_3将第3个float型本地变量推送至栈顶
0×26dload_0将第0个double型本地变量推送至栈顶
0×27dload_1将第1个double型本地变量推送至栈顶
0×28dload_2将第2个double型本地变量推送至栈顶
0×29dload_3将第3个double型本地变量推送至栈顶
0x2aaload_0将第0个引用类型本地变量推送至栈顶
0x2baload_1将第1个引用类型本地变量推送至栈顶
0x2caload_2将第2个引用类型本地变量推送至栈顶
0x2daload_3将第3个引用类型本地变量推送至栈顶
0x2eiaload将int型数组指定索引的值推送至栈顶
0x2flaload将long型数组指定索引的值推送至栈顶
0×30faload将float型数组指定索引的值推送至栈顶
0×31daload将double型数组指定索引的值推送至栈顶
0×32aaload将引用型数组指定索引的值推送至栈顶
0×33baload将boolean或byte型数组指定索引的值推送至栈顶
0×34caload将char型数组指定索引的值推送至栈顶
0×35saload将short型数组指定索引的值推送至栈顶
0×36istore将栈顶int型数值存入指定本地变量
0×37lstore将栈顶long型数值存入指定本地变量
0×38fstore将栈顶float型数值存入指定本地变量
0×39dstore将栈顶double型数值存入指定本地变量
0x3aastore将栈顶引用型数值存入指定本地变量
0x3bistore_0将栈顶int型数值存入第0个本地变量
0x3cistore_1将栈顶int型数值存入第1个本地变量
0x3distore_2将栈顶int型数值存入第2个本地变量
0x3eistore_3将栈顶int型数值存入第3个本地变量
0x3flstore_0将栈顶long型数值存入第0个本地变量
0×40lstore_1将栈顶long型数值存入第1个本地变量
0×41lstore_2将栈顶long型数值存入第2个本地变量
0×42lstore_3将栈顶long型数值存入第3个本地变量
0×43fstore_0将栈顶float型数值存入第0个本地变量
0×44fstore_1将栈顶float型数值存入第1个本地变量
0×45fstore_2将栈顶float型数值存入第2个本地变量
0×46fstore_3将栈顶float型数值存入第3个本地变量
0×47dstore_0将栈顶double型数值存入第0个本地变量
0×48dstore_1将栈顶double型数值存入第1个本地变量
0×49dstore_2将栈顶double型数值存入第2个本地变量
0x4adstore_3将栈顶double型数值存入第3个本地变量
0x4bastore_0将栈顶引用型数值存入第0个本地变量
0x4castore_1将栈顶引用型数值存入第1个本地变量
0x4dastore_2将栈顶引用型数值存入第2个本地变量
0x4eastore_3将栈顶引用型数值存入第3个本地变量
0x4fiastore将栈顶int型数值存入指定数组的指定索引位置
0×50lastore将栈顶long型数值存入指定数组的指定索引位置
0×51fastore将栈顶float型数值存入指定数组的指定索引位置
0×52dastore将栈顶double型数值存入指定数组的指定索引位置
0×53aastore将栈顶引用型数值存入指定数组的指定索引位置
0×54bastore将栈顶boolean或byte型数值存入指定数组的指定索引位置
0×55castore将栈顶char型数值存入指定数组的指定索引位置
0×56sastore将栈顶short型数值存入指定数组的指定索引位置
0×57pop将栈顶数值弹出 (数值不能是long或double类型的)
0×58pop2将栈顶的一个(long或double类型的)或两个数值弹出(其它)
0×59dup复制栈顶数值并将复制值压入栈顶
0x5adup_x1复制栈顶数值并将两个复制值压入栈顶
0x5bdup_x2复制栈顶数值并将三个(或两个)复制值压入栈顶
0x5cdup2复制栈顶一个(long或double类型的)或两个(其它)数值并将复制值压入栈顶
0x5ddup2_x1<待补充>
0x5edup2_x2<待补充>
0x5fswap将栈最顶端的两个数值互换(数值不能是long或double类型的)
0×60iadd将栈顶两int型数值相加并将结果压入栈顶
0×61ladd将栈顶两long型数值相加并将结果压入栈顶
0×62fadd将栈顶两float型数值相加并将结果压入栈顶
0×63dadd将栈顶两double型数值相加并将结果压入栈顶
0×64isub将栈顶两int型数值相减并将结果压入栈顶
0×65lsub将栈顶两long型数值相减并将结果压入栈顶
0×66fsub将栈顶两float型数值相减并将结果压入栈顶
0×67dsub将栈顶两double型数值相减并将结果压入栈顶
0×68imul将栈顶两int型数值相乘并将结果压入栈顶
0×69lmul将栈顶两long型数值相乘并将结果压入栈顶
0x6afmul将栈顶两float型数值相乘并将结果压入栈顶
0x6bdmul将栈顶两double型数值相乘并将结果压入栈顶
0x6cidiv将栈顶两int型数值相除并将结果压入栈顶
0x6dldiv将栈顶两long型数值相除并将结果压入栈顶
0x6efdiv将栈顶两float型数值相除并将结果压入栈顶
0x6fddiv将栈顶两double型数值相除并将结果压入栈顶
0×70irem将栈顶两int型数值作取模运算并将结果压入栈顶
0×71lrem将栈顶两long型数值作取模运算并将结果压入栈顶
0×72frem将栈顶两float型数值作取模运算并将结果压入栈顶
0×73drem将栈顶两double型数值作取模运算并将结果压入栈顶
0×74ineg将栈顶int型数值取负并将结果压入栈顶
0×75lneg将栈顶long型数值取负并将结果压入栈顶
0×76fneg将栈顶float型数值取负并将结果压入栈顶
0×77dneg将栈顶double型数值取负并将结果压入栈顶
0×78ishl将int型数值左移位指定位数并将结果压入栈顶
0×79lshl将long型数值左移位指定位数并将结果压入栈顶
0x7aishr将int型数值右(符号)移位指定位数并将结果压入栈顶
0x7blshr将long型数值右(符号)移位指定位数并将结果压入栈顶
0x7ciushr将int型数值右(无符号)移位指定位数并将结果压入栈顶
0x7dlushr将long型数值右(无符号)移位指定位数并将结果压入栈顶
0x7eiand将栈顶两int型数值作“按位与”并将结果压入栈顶
0x7fland将栈顶两long型数值作“按位与”并将结果压入栈顶
0×80ior将栈顶两int型数值作“按位或”并将结果压入栈顶
0×81lor将栈顶两long型数值作“按位或”并将结果压入栈顶
0×82ixor将栈顶两int型数值作“按位异或”并将结果压入栈顶
0×83lxor将栈顶两long型数值作“按位异或”并将结果压入栈顶
0×84iinc将指定int型变量增加指定值,可以有两个变量,分别表示index, const,index指第index个int型本地变量,const增加的值
0×85i2l将栈顶int型数值强制转换成long型数值并将结果压入栈顶
0×86i2f将栈顶int型数值强制转换成float型数值并将结果压入栈顶
0×87i2d将栈顶int型数值强制转换成double型数值并将结果压入栈顶
0×88l2i将栈顶long型数值强制转换成int型数值并将结果压入栈顶
0×89l2f将栈顶long型数值强制转换成float型数值并将结果压入栈顶
0x8al2d将栈顶long型数值强制转换成double型数值并将结果压入栈顶
0x8bf2i将栈顶float型数值强制转换成int型数值并将结果压入栈顶
0x8cf2l将栈顶float型数值强制转换成long型数值并将结果压入栈顶
0x8df2d将栈顶float型数值强制转换成double型数值并将结果压入栈顶
0x8ed2i将栈顶double型数值强制转换成int型数值并将结果压入栈顶
0x8fd2l将栈顶double型数值强制转换成long型数值并将结果压入栈顶
0×90d2f将栈顶double型数值强制转换成float型数值并将结果压入栈顶
0×91i2b将栈顶int型数值强制转换成byte型数值并将结果压入栈顶
0×92i2c将栈顶int型数值强制转换成char型数值并将结果压入栈顶
0×93i2s将栈顶int型数值强制转换成short型数值并将结果压入栈顶
0×94lcmp比较栈顶两long型数值大小,并将结果(1,0,-1)压入栈顶
0×95fcmpl比较栈顶两float型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将-1压入栈顶
0×96fcmpg比较栈顶两float型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将1压入栈顶
0×97dcmpl比较栈顶两double型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将-1压入栈顶
0×98dcmpg比较栈顶两double型数值大小,并将结果(1,0,-1)压入栈顶;当其中一个数值为NaN时,将1压入栈顶
0×99ifeq当栈顶int型数值等于0时跳转
0x9aifne当栈顶int型数值不等于0时跳转
0x9biflt当栈顶int型数值小于0时跳转
0x9cifge当栈顶int型数值大于等于0时跳转
0x9difgt当栈顶int型数值大于0时跳转
0x9eifle当栈顶int型数值小于等于0时跳转
0x9fif_icmpeq比较栈顶两int型数值大小,当结果等于0时跳转
0xa0if_icmpne比较栈顶两int型数值大小,当结果不等于0时跳转
0xa1if_icmplt比较栈顶两int型数值大小,当结果小于0时跳转
0xa2if_icmpge比较栈顶两int型数值大小,当结果大于等于0时跳转
0xa3if_icmpgt比较栈顶两int型数值大小,当结果大于0时跳转
0xa4if_icmple比较栈顶两int型数值大小,当结果小于等于0时跳转
0xa5if_acmpeq比较栈顶两引用型数值,当结果相等时跳转
0xa6if_acmpne比较栈顶两引用型数值,当结果不相等时跳转
0xa7goto无条件跳转
0xa8jsr跳转至指定16位offset位置,并将jsr下一条指令地址压入栈顶
0xa9ret返回至本地变量指定的index的指令位置(一般与jsr, jsr_w联合使用)
0xaatableswitch用于switch条件跳转,case值连续(可变长度指令)
0xablookupswitch用于switch条件跳转,case值不连续(可变长度指令)
0xacireturn从当前方法返回int
0xadlreturn从当前方法返回long
0xaefreturn从当前方法返回float
0xafdreturn从当前方法返回double
0xb0areturn从当前方法返回对象引用
0xb1return从当前方法返回void
0xb2getstatic获取指定类的静态域,并将其值压入栈顶
0xb3putstatic为指定的类的静态域赋值
0xb4getfield获取指定类的实例域,并将其值压入栈顶
0xb5putfield为指定的类的实例域赋值
0xb6invokevirtual调用实例方法
0xb7invokespecial调用超类构造方法,实例初始化方法,私有方法
0xb8invokestatic调用静态方法
0xb9invokeinterface调用接口方法
0xba
0xbbnew创建一个对象,并将其引用值压入栈顶
0xbcnewarray创建一个指定原始类型(如int, float, char…)的数组,并将其引用值压入栈顶
0xbdanewarray创建一个引用型(如类,接口,数组)的数组,并将其引用值压入栈顶
0xbearraylength获得数组的长度值并压入栈顶
0xbfathrow将栈顶的异常抛出
0xc0checkcast检验类型转换,检验未通过将抛出ClassCastException
0xc1instanceof检验对象是否是指定的类的实例,如果是将1压入栈顶,否则将0压入栈顶
0xc2monitorenter获得对象的锁,用于同步方法或同步块
0xc3monitorexit释放对象的锁,用于同步方法或同步块
0xc4wide当本地变量的索引超过255时使用该指令扩展索引宽度。
0xc5multianewarraycreate a new array of dimensions dimensions with elements of type identified by class reference in constant pool index (indexbyte1 << 8 + indexbyte2); the sizes of each dimension is identified by count1, [count2, etc.]
0xc6ifnullif value is null, branch to instruction at branchoffset (signed short constructed from unsigned bytes branchbyte1 << 8 + branchbyte2)
0xc7ifnonnullif value is not null, branch to instruction at branchoffset (signed short constructed from unsigned bytes branchbyte1 << 8 + branchbyte2)
0xc8goto_wgoes to another instruction at branchoffset (signed int constructed from unsigned bytes branchbyte1 << 24 + branchbyte2 << 16 + branchbyte3 << 8 + branchbyte4)
0xc9jsr_wjump to subroutine at branchoffset (signed int constructed from unsigned bytes branchbyte1 << 24 + branchbyte2 << 16 + branchbyte3 << 8 + branchbyte4) and place the return address on the stack
0xcabreakpointreserved for breakpoints in Java debuggers; should not appear in any class file
0xcb-0xfd未命名these values are currently unassigned for opcodes and are reserved for future use
0xfeimpdep1reserved for implementation-dependent operations within debuggers; should not appear in any class file
0xffimpdep2reserved for implementation-dependent operations within debuggers; should not appear in any class file

有了以上指令集表,那么在查看字节码就方便多了。

对应英文 => en.wikipedia.org/wiki/Java_b…

来几个sample:

sample1

  • java
public void sample1(){
            int num = 5;
            }
        
  • javap -c 查看字节码
public void sample1();
            Code:
               0: iconst_5
               1: istore_1
               2: return
               
  • 解释
iconst_5    //将int型5推送至栈顶
        istore_1    //将栈顶int型数值存入第1个本地变量
        return    //从当前方法返回void
        

sample2

  • java
public int sample2(int a, int b) {
            return a + b;
            }
        
  • 字节码及解释
public int sample2(int, int);
            Code:
               0: iload_1    //将第1个int型本地变量推送至栈顶
               1: iload_2    //将第2个int型本地变量推送至栈顶
               2: iadd       //将栈顶两int型数值相加并将结果压入栈顶
               3: ireturn    //从当前方法返回int
               

sample3

  • java

稍稍复杂点

public float sample3() {
            float num = 0;
            for (int i = 0; i < 5; i++) {
                num *= i;
            }
            return num;
            }
        
  • 字节码及解释
public float sample3();
            Code:
               0: fconst_0    //将float型0推送至栈顶
               1: fstore_1    //将栈顶float型数值存入第1个本地变量
               2: iconst_0    //将int型0推送至栈顶,也就是for循环中的i = 0
               3: istore_2    //将栈顶int型数值存入第2个本地变量
               4: iload_2     //将第2个int型本地变量推送至栈顶
               5: iconst_5    //将int型5推送至栈顶,也就是for循环中的 最大值5 
               6: if_icmpge     20    //比较栈顶两int型数值大小,当结果大于等于0时跳转,
               //也就是比较0是否大于等于5,(cpmge指compare larger equals),如果是跳转到到第20条指令
               9: fload_1    //将第1个float型本地变量推送至栈顶,也就是变量num
              10: iload_2    //将第2个int型本地变量推送至栈顶,也就是for循环中的变量i
              11: i2f        //int型强转为float型,也就是把变量i强转成float
              12: fmul       //将栈顶两float型数值相乘并将结果压入栈顶,也就是i与num相乘
              13: fstore_1    //将栈顶float型数值存入第1个本地变量,也就是之前i与num的乘积
              14: iinc          2, 1    //将指定int型变量增加指定值,将第2个int型本地变量增加1,
              //可以看到,第2个int型本地变量就是之前的变量i
              17: goto          4    //无条件跳转到指令4,实现循环效果
              20: fload_1      //将第1个float型本地变量推送至栈顶
              21: freturn    //从当前方法返回float
              

以上是三个比较简单的例子,读者看懂后可以从字节码为切入点,去着手分析下为何switch比if else效率高。

    原文作者:JVM
    原文地址: https://juejin.im/entry/588085221b69e60059035f0a
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞