参考《深入理解JVM》
1 生成类文件
编写java程序,通过编译器编译或者使用javac编译器形成字节码文件(*.class)。如下代码:
public class TestClass {
private int m;
public int inc() {
return m + 1;
}
}
使用idea进行编译,在out目录下生成TestClass.class文件。后续使用notepad++通过16进制查看该文件。
2 Class类文件的结构
Class文件格式采用结构体来存储数据,伪结构只有两种数据类型:“无符号数"和"表”。
- 无符号数属于基本的数据类型,以u1、u2、u4、u8来分别表示1个字节、2个字节、3个字节、4个字节和8个字节的无符号数,无符号数可以用来描述数字、索引引用、数量值或按照UTF-8编码构成字符串值。
- 表是由多个无符号数或其他表作为数据项构成的复合数据类型。整个class文件可以视作一张表如下图所示:
2.1 魔数与Class文件的版本
magic、minor_version、major_version对应下图三个框。
magic(魔数):确定这个文件是否为一个能被虚拟机接受的Class文件。
minor_version(次版本号):短暂使用过,基本置为0。
major_version(主版本号):高版本的jdk能向下兼容以前版本的class文件。反之不行。
2.2 常量池
通过下图可以看到,constant_pool_count的个数为(0x16 - 1) 22 - 1 = 21个。(规定索引计数从1开始,不引用则索引设置为0)。在constant_pool_count之后跟的是常量池。发现tag字段是(0x0a)10,查表发现之后跟的是两个2字节的index。
使用javap -v TestClass.class
对字节码文件进行反编译发现常量池第一条正好对应。如下: