美文网首页
JVM_字节码:字节码常量池分析

JVM_字节码:字节码常量池分析

作者: CalmHeart | 来源:发表于2019-06-27 00:12 被阅读0次

分析的条件: 基于上一篇的16进制字节码及反编译结果(javap -verbose the desire class)

分析的时候可以使用(jclasslib)进行查看,也提供了IDEA的plugin,显示的信息比较完善,比较优雅的分门别类字节码的相关信息。

IDEA插件界面
分析的结果

反编译结果:

Classfile /C:/spring_lecture/target/classes/com/compass/spring_lecture/binarycode/MyTest1.class
  Last modified 2019年6月26日; size 521 bytes
  MD5 checksum e84ac7b4ed245fd5824849fb69bacc27
  Compiled from "MyTest1.java"
public class com.compass.spring_lecture.binarycode.MyTest1
  minor version: 0 //小版本
  major version: 52 //大版本
  flags: (0x0021) ACC_PUBLIC, ACC_SUPER
  this_class: #3                          // com/compass/spring_lecture/binarycode/MyTest1
  super_class: #4                         // java/lang/Object
  interfaces: 0, fields: 1, methods: 3, attributes: 1
Constant pool://常量池
   #1 = Methodref          #4.#20         // java/lang/Object."<init>":()V
   #2 = Fieldref           #3.#21         // com/compass/spring_lecture/binarycode/MyTest1.a:I
   #3 = Class              #22            // com/compass/spring_lecture/binarycode/MyTest1
   #4 = Class              #23            // java/lang/Object
   #5 = Utf8               a
   #6 = Utf8               I
   #7 = Utf8               <init>
   #8 = Utf8               ()V
   #9 = Utf8               Code
  #10 = Utf8               LineNumberTable
  #11 = Utf8               LocalVariableTable
  #12 = Utf8               this
  #13 = Utf8               Lcom/compass/spring_lecture/binarycode/MyTest1;
  #14 = Utf8               getA
  #15 = Utf8               ()I
  #16 = Utf8               setA
  #17 = Utf8               (I)V
  #18 = Utf8               SourceFile
  #19 = Utf8               MyTest1.java
  #20 = NameAndType        #7:#8          // "<init>":()V
  #21 = NameAndType        #5:#6          // a:I
  #22 = Utf8               com/compass/spring_lecture/binarycode/MyTest1
  #23 = Utf8               java/lang/Object
{
  public com.compass.spring_lecture.binarycode.MyTest1();
    descriptor: ()V
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=2, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: aload_0
         5: iconst_1
         6: putfield      #2                  // Field a:I
         9: return
      LineNumberTable:
        line 4: 0
        line 6: 4
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      10     0  this   Lcom/compass/spring_lecture/binarycode/MyTest1;

  public int getA();
    descriptor: ()I
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: getfield      #2                  // Field a:I
         4: ireturn
      LineNumberTable:
        line 10: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Lcom/compass/spring_lecture/binarycode/MyTest1;

  public void setA(int);
    descriptor: (I)V
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: iload_1
         2: putfield      #2                  // Field a:I
         5: return
      LineNumberTable: //行号表
        line 14: 0
        line 15: 5
      LocalVariableTable: //局部变量表
        Start  Length  Slot  Name   Signature
            0       6     0  this   Lcom/compass/spring_lecture/binarycode/MyTest1;
            0       6     1     a   I
}
SourceFile: "MyTest1.java"

通过16进制编辑器查看字节码结果:


16进制表示

  • 常量池(constant pool):要和java中的常量区分开
    紧接着主版本号之后的就是常量池入口,常量池的长度是不确定的,一个java类中定义的很多信息都是由常量池来维护和描述的,可以将常量池看作是class文件的资源仓库,比如说java类中定义的方法与变量信息,都是存储在常量池中的,常量池中主要存储两类常量:字面量和符号引用[符号引用转换成直接引用]。字面量比如说文本字符串,java中声明为final的常量值等。而符号引用如类和接口的全局限定名,字段的名称和描述符,方法的名称和描述符等。

  • 常量池的总体结构:
    java类对应的常量池主要由常量池数量和常量池数组(表结构)这2部分共同组成,可以最大限度地压缩字节码的体积,常量池数量紧跟在主版本后面,占据2个字节,常量池数组紧跟在常量池数量之后。常量池数组与一般的数组不同的是,常量池数组中不同的元素的类型、结构都是不同的。长度当然也就不同,但是每一种元素的第一个数据都是一个u1类型,该字节是一个标志位,占据一个1个字节,JVM在解析常量池的时候,会根据这个u1类型来获取对应元素的具体类型。分析字节码:00 18 共有24个常量,反编译字节码中只有23个。为什么呢? 值得注意的是:常量池中元素的个数=常量池数-1(其中0暂时不使用)。目的是满足某些常量池索引值的数据在特定的情况下需要表达[不引用任何的常量池]的含义,根本原因在于,索引为0也是一个常量(保留常量),只不过它不位于常量表中。这个常量就对应null值,索引常量池的索引从1而非0开始。


字节码的数据类型:共有11种数据类型

Class的数据类型

相关文章

网友评论

      本文标题:JVM_字节码:字节码常量池分析

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