发布时间:2022-10-26 文章分类:编程知识 投稿人:赵颖 字号: 默认 | | 超大 打印
目录
  • 1 Class类文件结构
  • 2 常量池

    • 2.1 字面量
    • 2.2 符号引用
    • 2.3 常量的表结构

      • 2.3.1 表结构的属性解析
      • 2.3.2 17种常量的表结构解析
    • 2.4 javap分析字节码文件

1 Class类文件结构

  1. Class文件:是一组以8个字节为基础单位的二进制流,当需要存储空间超过8字节时,会按照高位在前的方式分割【大端法】成若干个8个字节进行存储
  2. 魔数:头4个字节,固定值为:0xCAFEBABE
  3. 版本号:魔数的后继4个字节,第5和第6个字节是次版本号,第7和第8个字节是主版本号
  4. 常量池:版本号的后续字节。
  5. 常量池的常量计数值:常量池部分的前2个字节,表示该文件有多少个常量,计数值从1开始
  6. 常量池的常量:分为字面量、符号引用
  7. 字面量:接近于Java语言层面的常量概念,如文本字符串、被声明为final的常量值等
  8. 符号引用:编译原理方面的概念。
    5 字节码文件结构

2 常量池

常量池的常量分为两种类型:字面量、符号引用

2.1 字面量

字面量比 较接近于Java语言层面的常量概念,如文本字符串、被声明为final的常量值等。

2.2 符号引用

符号引用则属于编译原理方面的概念,包括这几类:

2.3 常量的表结构

常量池中每一项常量都是一个表,一共有17种表结构,因此一共有17种常量。17种常量类型如下所示:
5 字节码文件结构
5 字节码文件结构

2.3.1 表结构的属性解析

表结构,可以表示为:【属性:对应的值】,每种常量类型(表)都有一个或者多个属性,所有属性的解释如下:

2.3.2 17种常量的表结构解析

表结构起始的第一位都是flag标志位
5 字节码文件结构
5 字节码文件结构

(JDK7增加的三种:CONSTANT_MethodHandle_info、CONSTANT_MethodType_info和 CONSTANT_InvokeDynamic_info,JDK11中又增加了第四种常量CONSTANT_Dynamic_info,在后续章节中详细解)

2.4 javap分析字节码文件

javap:用于分析Class文件字节码的工具,在JDK的bin目录中。
javap -verbose 输出字节码的内容。

测试的java代码

package com.minnesota.practice.test;
public class ByteCodeTest {
    private static final String DREAM = "byte dancing";
    public static void main(String[] args) {
        int a = 1;
        int b = 2;
        int c = a+b;
    }
}

ByteCodeTest.class 内容浏览:
5 字节码文件结构

javap -verbose ByteCodeTest.class 解析后

javap -verbose ByteCodeTest.class
输出结果:
Classfile /路径脱敏/ByteCodeTest.class
  Last modified 2022-10-25; size 396 bytes
  MD5 checksum be72a2ed3bf8578da63968ce35880d3d
  Compiled from "ByteCodeTest.java"
public class com.minnesota.practice.test.ByteCodeTest
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #3.#16         // java/lang/Object."<init>":()V
   #2 = Class              #17            // com/minnesota/practice/test/ByteCodeTest
   #3 = Class              #18            // java/lang/Object
   #4 = Utf8               DREAM
   #5 = Utf8               Ljava/lang/String;
   #6 = Utf8               ConstantValue
   #7 = String             #19            // byte dancing
   #8 = Utf8               <init>
   #9 = Utf8               ()V
  #10 = Utf8               Code
  #11 = Utf8               LineNumberTable
  #12 = Utf8               main
  #13 = Utf8               ([Ljava/lang/String;)V
  #14 = Utf8               SourceFile
  #15 = Utf8               ByteCodeTest.java
  #16 = NameAndType        #8:#9          // "<init>":()V
  #17 = Utf8               com/minnesota/practice/test/ByteCodeTest
  #18 = Utf8               java/lang/Object
  #19 = Utf8               byte dancing
{
  public com.minnesota.practice.test.ByteCodeTest();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0
  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=4, args_size=1
         0: iconst_1
         1: istore_1
         2: iconst_2
         3: istore_2
         4: iload_1
         5: iload_2
         6: iadd
         7: istore_3
         8: return
      LineNumberTable:
        line 6: 0
        line 7: 2
        line 8: 4
        line 9: 8
}
SourceFile: "ByteCodeTest.java"

todo 这块还没看懂

待续!