文章目录
前言
上一篇博客 【Java 虚拟机原理】Class 字节码二进制文件分析 三 ( 访问和修饰标志 | 类索引 | 父类索引 | 接口计数器 | 接口表 | 字段计数器 | 字段表 ) 分析了常量表之后的一些数据 ;
分析到 Student.class 字节码文件的字段表数据 ,
本篇博客中 , 继续向后分析 字节码对应数据 ;
分析的原始数据是 【Java 虚拟机原理】Class 字节码二进制文件分析 一 ( 字节码文件附加信息 | 魔数 | 次版本号 | 主版本号 | 常量池个数 ) 二、字节码文件示例 章节中的 Java 源码 , Class 字节码 , 字节码附加信息 ;
public class Student {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
字节码附加信息 :
D:\jvm>javap -v Student.class
Classfile /D:/jvm/Student.class
Last modified 2021-9-4; size 392 bytes
MD5 checksum 8b9bb897bb8cf2a8addf04be5b7b915f
Compiled from "Student.java"
public class Student
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #4.#17 // java/lang/Object."<init>":()V
#2 = Fieldref #3.#18 // Student.name:Ljava/lang/String;
#3 = Class #19 // Student
#4 = Class #20 // java/lang/Object
#5 = Utf8 name
#6 = Utf8 Ljava/lang/String;
#7 = Utf8 <init>
#8 = Utf8 ()V
#9 = Utf8 Code
#10 = Utf8 LineNumberTable
#11 = Utf8 getName
#12 = Utf8 ()Ljava/lang/String;
#13 = Utf8 setName
#14 = Utf8 (Ljava/lang/String;)V
#15 = Utf8 SourceFile
#16 = Utf8 Student.java
#17 = NameAndType #7:#8 // "<init>":()V
#18 = NameAndType #5:#6 // name:Ljava/lang/String;
#19 = Utf8 Student
#20 = Utf8 java/lang/Object
{
public Student();
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 1: 0
public java.lang.String getName();
descriptor: ()Ljava/lang/String;
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: getfield #2 // Field name:Ljava/lang/String;
4: areturn
LineNumberTable:
line 5: 0
public void setName(java.lang.String);
descriptor: (Ljava/lang/String;)V
flags: ACC_PUBLIC
Code:
stack=2, locals=2, args_size=2
0: aload_0
1: aload_1
2: putfield #2 // Field name:Ljava/lang/String;
5: return
LineNumberTable:
line 9: 0
line 10: 5
}
SourceFile: "Student.java"
一、字段表总数据结构
在 fields_count ( 字段计数器 ) 后面就是 字段表 , 字段表可以是 1 1 1 个字段信息 , 也可以是若干个 字段信息 ;
下表表示单个字段信息 , 如果有多个字段信息 , 就会依次进行排列 ;
如下图红色矩形框中显示 :
单个 field_info 字段信息结构如下 : u2 表示 2 2 2 个字节 ;
二、访问标志
字段表 前
2
2
2 字节表示 " 访问标志 " , 00 02
;
参考 下面的 字段访问标志 解读 00 02
标志 ;
00 02
表示当前字段是 private 私有字段 ;
三、字段名称
name_index 占 2 2 2 字节 , 指向常量池中的 UTF-8 字符串应用 ;
00 05
表示常量池中的 #5
引用 , 表示字段名称是 name
;
Constant pool:
#5 = Utf8 name
四、字段描述符
descriptor_index 字段描述符 , 表示字段的类型 , 占 2 2 2 字节 , 指向常量池索引 ;
00 06
表示 字段描述符 是常量池中的 #6
,
#6 = Utf8 Ljava/lang/String;
五、属性项目数
attributes_count 占 2 2 2 字节 , 表示属性包含的项目数 ;
这里的属性项目数为 00 00
, 那么说明后续属性项目个数为
0
0
0 , 字段表中本字段信息到此为止 ; 开始显示下一个字段 ;
当前的字节码文件中只有 1 1 1 个字段 , 字段表显示 1 1 1 个字段结束后 , 后面的字节是 方法计数器 和 方法表 的数据 ;