美文网首页
字节码指令

字节码指令

作者: sunpy | 来源:发表于2018-09-25 13:57 被阅读31次

介绍

jvm字节码指令就是由一组带有特定操作含义的数字并且后面跟上操作参数组成。

加载和存储指令

将数据从栈桢中的局部变量表和操作数栈之间来回传输。
局部变量表:存放编译期就已经知道的各种基本数据类型、对象引用类型;注意:在编译期间完成对所需的内存空间的分配。
操作数栈:就是一个先进后出的栈,在方法执行的时候会将各种字节码指令操作栈(譬如入栈和出栈等)。


  1. 局部变量加载到操作数栈
    iload 、 iload_<n> 、lload 、lload_<n> 、fload 、 fload_<n> 、 dload dload_<n>、 aload 、 aload_<n>

  1. 从操作数栈存储到局部变量表
    istore 、istore_<n> 、 lstore 、 lstore_<n>、 fstore 、fstore_<n> dstore 、 dstore_<n> 、astore 、 astore_<n>

  1. 常量加载到操作数栈
    bipush、 sipush 、 ldc、 ldc_w、 ldc2_w 、aconst_null 、 iconst_null iconst_<i>、 lconst_<l> 、 fconst_<f> 、dconst_<d>

运算指令

就是对操作数栈中栈顶上的值进行运算,然后将结果入栈。


  1. 加法指令
    iadd、 ladd、 fadd、 dadd

  1. 减法指令
    isub、 lsub 、 fusb、 dsub

  1. 乘法指令
    imul 、 lmul 、fmul 、 dmul

  1. 除法指令
    idiv、 ldiv、 fdiv 、 ddiv

  1. 求余指令
    irem、 lrem、 frem、 drem

  1. 取反指令
    ineg、 lneg 、 fneg 、 dneg

  1. 位移指令
    ishl、 ishr、 iushr、 lshl 、 lshr 、 lushr

  1. 按位或指令
    ior、 lor

  1. 按位与指令
    iand 、 land

  1. 按位异或指令
    ixor 、 lxor

  1. 局部变量自增指令
    iinc

  1. 比较指令
    dcmpg 、 dcmpl 、fcmpg 、 fcmpl、 lcmp

类型转换指令

  1. 宽化类型转换
    譬如:int类型到long、float或者double类型;float类型到double类型;long类型到float、double类型。
    i2l、f2b、l2f、l2d、f2d

  1. 窄化类型转换
    i2b、i2c、i2s、l2i、f2i、f2l、d2i、d2l、d2f

对象创建与访问指令

  1. 创建类实例的指令
    new

  1. 创建数组的指令
    newarray、anewarray、multianewarray

  1. 访问static类型的字段
    getstatic、putstatic

  1. 访问非static类型的字段
    getfield、putfield

  1. 把一个数组元素加载到操作数栈
    baload、caload、saload、iaload、laload、faload、daload、aaload

  1. 将一个操作数栈的值存储到数组元素中的指令
    bastore、castore、sastore、iastore、fastore、dastore、aastore

  1. 取数组长度的指令
    arraylength

  1. 检查类实例类型的指令
    instanceof、checkcast

操作数栈管理指令

  1. 将操作数栈顶的元素一个或两个出栈
    pop、pop2

  1. 复制一个栈顶元素或者两个栈顶元素重新入栈
    dup、dup2、dup_x1、dup2_x1、dup_x2、dup2_x2

  1. 将栈最顶端的两个元素互换
    swap

控制转移指令

  1. 条件分支
    ifeq、iflt、ifle、ifne、ifgt、ifge、ifnull、ifnonnull、if_icmpeq、if_icmpne、if_icmplt、if_icmpgt、if_icmple、if_icmpge、if_acmpeq、if_acmpne

  1. 复合条件分支
    tableswitch、lookupswitch

  1. 无条件分支
    goto、goto_w、jsr、jsr_w、ret

方法调用和返回指令

  1. 调用对象的实例方法
    **invokevirtual **

  1. 用于调用接口方法
    invokeinterface

  1. 用于调用一些需要特殊处理的实例方法,包括实例初始化(<init>)方法、私有方法和父类方法
    invokespecial

  1. 调用静态方法
    invokestatic

  1. 运行时动态解析引用方法,动态调用
    invokedynamic 指令用于在运行时动态解析出调用点限定符所引用的方法,并执行该方法

异常处理指令

athrow


同步指令

Java虚拟机的指令集中有monitorenter、monitorexit两条指令来支持synchronized关键字的语义


例子

求和功能例子:

public class MyByteCodeTest {

    public static void main(String[] args) {
        int sum = 0;
        
        for (int i = 0 ; i <= 10 ; i++) {
            sum += i;
        }
        
        System.out.println(sum);
    }
}

javap -verbose MyByteCodeTest.class查看:

C:\Users\Administrator\Desktop>javap -verbose MyByteCodeTest.class
Classfile /C:/Users/Administrator/Desktop/MyByteCodeTest.class
  Last modified 2018-9-25; size 648 bytes
  MD5 checksum 757fc39abe1522912132c7aaf462a0bb
  Compiled from "MyByteCodeTest.java"
public class cn.spy.spring.test.MyByteCodeTest
  minor version: 0
  major version: 51
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Class              #2             // cn/spy/spring/test/MyByteCodeTest
   #2 = Utf8               cn/spy/spring/test/MyByteCodeTest
   #3 = Class              #4             // java/lang/Object
   #4 = Utf8               java/lang/Object
   #5 = Utf8               <init>
   #6 = Utf8               ()V
   #7 = Utf8               Code
   #8 = Methodref          #3.#9          // java/lang/Object."<init>":()V
   #9 = NameAndType        #5:#6          // "<init>":()V
  #10 = Utf8               LineNumberTable
  #11 = Utf8               LocalVariableTable
  #12 = Utf8               this
  #13 = Utf8               Lcn/spy/spring/test/MyByteCodeTest;
  #14 = Utf8               main
  #15 = Utf8               ([Ljava/lang/String;)V
  #16 = Fieldref           #17.#19        // java/lang/System.out:Ljava/io/Print
Stream;
  #17 = Class              #18            // java/lang/System
  #18 = Utf8               java/lang/System
  #19 = NameAndType        #20:#21        // out:Ljava/io/PrintStream;
  #20 = Utf8               out
  #21 = Utf8               Ljava/io/PrintStream;
  #22 = Methodref          #23.#25        // java/io/PrintStream.println:(I)V
  #23 = Class              #24            // java/io/PrintStream
  #24 = Utf8               java/io/PrintStream
  #25 = NameAndType        #26:#27        // println:(I)V
  #26 = Utf8               println
  #27 = Utf8               (I)V
  #28 = Utf8               args
  #29 = Utf8               [Ljava/lang/String;
  #30 = Utf8               sum
  #31 = Utf8               I
  #32 = Utf8               i
  #33 = Utf8               StackMapTable
  #34 = Utf8               SourceFile
  #35 = Utf8               MyByteCodeTest.java
{
  public cn.spy.spring.test.MyByteCodeTest();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #8                  // Method java/lang/Object."<init>
":()V
         4: return
      LineNumberTable:
        line 3: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lcn/spy/spring/test/MyByteCodeTest;

  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=3, args_size=1
         0: iconst_0
         1: istore_1
         2: iconst_0
         3: istore_2
         4: goto          14
         7: iload_1
         8: iload_2
         9: iadd
        10: istore_1
        11: iinc          2, 1
        14: iload_2
        15: bipush        10
        17: if_icmple     7
        20: getstatic     #16                 // Field java/lang/System.out:Ljav
a/io/PrintStream;
        23: iload_1
        24: invokevirtual #22                 // Method java/io/PrintStream.prin
tln:(I)V
        27: return
      LineNumberTable:
        line 7: 0
        line 9: 2
        line 10: 7
        line 9: 11
        line 13: 20
        line 14: 27
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      28     0  args   [Ljava/lang/String;
            2      26     1   sum   I
            4      16     2     i   I
      StackMapTable: number_of_entries = 2
        frame_type = 253 /* append */
          offset_delta = 7
          locals = [ int, int ]
        frame_type = 6 /* same */
}
SourceFile: "MyByteCodeTest.java"

分析main函数中的字节码指令:

 0: iconst_0    // 简单的数值类型int(0)送到栈顶 -- int sum = 0;将sum=0这个int值压入栈顶
 1: istore_1    // 将栈顶int型数值存入第1个本地变量 -- 将sum=0这个int值存入第1个本地变量中
 2: iconst_0    // 简单的数值类型int(0)送到栈顶 -- int i = 0;将i=0这个int值压入栈顶
 3: istore_2    // 将栈顶int型数值存入第2个本地变量 -- 将i=0这个int值存入第2个本地变量中
 4: goto          14    //  无条件跳转到14行  -- 无条件跳转到14行代码
 7: iload_1     // 将第1个int型本地变量推送至栈顶
 8: iload_2     // 将第2个int型本地变量推送至栈顶
 9: iadd        // 将栈顶两int型数值相加并将结果压入栈顶
10: istore_1    // 将栈顶int型数值存入第1个本地变量
11: iinc          2, 1      // 将int型的局部变量自增
14: iload_2     // 将第2个int型本地变量推送至栈顶
15: bipush        10    // 将单字节的常量值(-128~127)推送至栈顶
17: if_icmple     7     // 比较栈顶两int型数值的大小,当结果小于等于0时跳转
20: getstatic     #16                 // Field java/lang/System.out:Ljava/io/PrintStream;
23: iload_1     // 将第1个int型本地变量推送至栈顶
24: invokevirtual #22                 // Method java/io/PrintStream.println:(I)V
27: return      //方法返回指令

分析结果:

1. int sum = 0;将sum=0这个int值压入栈顶
2. 将sum=0这个int值存入第1个本地变量中
3. int i = 0;将i=0这个int值压入栈顶
4. 将i=0这个int值存入第2个本地变量中
5. 跳转到14行代码
6. 将第2个本地变量中i=0压入栈顶
7. 将常量值10压入栈顶
8. 栈顶值i=0小于栈顶常量10,将跳转至第7行代码
9. 将第1个本地变量中sum=0压入栈顶
10. 将第2个本地变量中i=0压入栈顶
11. 将栈顶sum=0和i=0两个值相加,结果再次压入栈顶
12. 将上次相加结果值存入第1个本地变量中
13. 将第2个本地变量自增
14. 将第2个本地变量压入栈顶
15. 然后又继续执行第7步,步骤7到步骤15开始循环直到i自增到不小于10
16. 获取类变量(PrintStream)
17. 将第1个本地变量中的值sum压入栈中
18. 调用对象的实例方法println
19. 返回值为void的函数返回指令

总结

可以发现jvm中的字节码指令对于for循环的实现采用if_icmple指令来实现的。而函数的处理采用操作数栈和本地变量表来进行实现的。

相关文章

  • Java 字节码指令

    字节码指令链接

  • JVM-06

    switch-case的字节码指令: Java代码如下: 字节码指令如下: 结论是:switch-case 语句 ...

  • Java虚拟机-字节码指令

    1 字节码指令 Java字节码指令的执行离不开操作数栈,局部变量表,和常量池。 2 常量池 对于字节码指定来说,常...

  • Java虚拟机知识点【字节码】

    字节码指令   Java虚拟机的字节码指令由一个字节长度,代表着某种特定操作含义的操作码以及跟随其后的零至多个代表...

  • 字节码指令

    介绍 jvm字节码指令就是由一组带有特定操作含义的数字并且后面跟上操作参数组成。 加载和存储指令 将数据从栈桢中的...

  • 字节码指令

    指令由一个字节码长度的、操作码(代表某种特定操作含义的数字)以及操作数(紧跟操作码后的0到多个参数)构成。 Jav...

  • 字节码指令

    描述 本文摘自深入理解Java虚拟机中关于字节码的介绍,部分指令参考,oracle字节码指令集。 数据类型:byt...

  • Java ByteCode

    什么是Java字节码指令?简而言之,Java字节码指令就是Java虚拟机能够听得懂、可执行的指令,可以说是Jvm层...

  • Java并发机制的底层原理

    Java程序执行:Java代码→Java字节码→字节码被类加载器加载到JVM里,JVM执行字节码→转化为汇编指令在...

  • 图解jvm--(三)类加载与字节码技术

    类加载与字节码技术 1.类文件结构 根据 JVM 规范,类文件结构如下 2.字节码指令 指令作用iconst_1i...

网友评论

      本文标题:字节码指令

      本文链接:https://www.haomeiwen.com/subject/inrzvftx.html