目录
基本概念
1.编译成字节码文件(.class)并不是只能Java能整
2.Class文件包含了三大类,两大数据结构
三大类:虚拟机指令,符号表,其他辅助信息
两大数据结构:无符号数,表(_info,类似一个结构)
3.计算机以010101010方式来执行。
4.类文件字段类型:
u1表示一个字节,u2表示二个字节,u4表示四个字节,u8表示八个字节
_info类型是表数据结构,对照相应的表进行分析,例如:常量池(cp_info)的分析,就对应着Class文件结构-常量池表
字节码的解释
常量池
类的元数据:描述类的数据。
1.魔数和版本
2.常量池计数器:两个字节表示(00 16)
十六进制转十进制:例如 16 = 6*16^0 + 1* 16^1 = 6 + 16 = 22
也就是有22个常量,但是实际上只有21个.留了一个空位
3.对照常量池表分析Class文件结构
(1)使用javap -v 命令(反汇编命令),对照常量池结构表分析对应的Class文件的常量即可;
(2)简单的字节码指令
(3)复杂一点的字节码
--解读虚拟机指令(invokespecial)
public class jvm.classFile.Test1
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #3.#13 // java/lang/Object."<init>":()V
#2 = Class #14 // jvm/classFile/Test1
#3 = Class #15 // java/lang/Object
#4 = Utf8 <init>
#5 = Utf8 ()V
#6 = Utf8 Code
#7 = Utf8 LineNumberTable
#8 = Utf8 LocalVariableTable
#9 = Utf8 this
#10 = Utf8 Ljvm/classFile/Test1;
#11 = Utf8 SourceFile
#12 = Utf8 Test1.java
#13 = NameAndType #4:#5 // "<init>":()V
#14 = Utf8 jvm/classFile/Test1
#15 = Utf8 java/lang/Object
{
public jvm.classFile.Test1();
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 7: 0
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Ljvm/classFile/Test1;
}
SourceFile: "Test1.java"
复杂的
public class jvm.classFile.Test2 extends jvm.classFile.Test1 implements java.io.Serializable,java.lang.Comparable
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #3.#24 // jvm/classFile/Test1."<init>":()V
#2 = Class #25 // jvm/classFile/Test2
#3 = Class #26 // jvm/classFile/Test1
#4 = Class #27 // java/io/Serializable
#5 = Class #28 // java/lang/Comparable
#6 = Utf8 name
#7 = Utf8 Ljava/lang/String;
#8 = Utf8 age
#9 = Utf8 I
#10 = Utf8 <init>
#11 = Utf8 ()V
#12 = Utf8 Code
#13 = Utf8 LineNumberTable
#14 = Utf8 LocalVariableTable
#15 = Utf8 this
#16 = Utf8 Ljvm/classFile/Test2;
#17 = Utf8 mathod
#18 = Utf8 compareTo
#19 = Utf8 (Ljava/lang/Object;)I
#20 = Utf8 o
#21 = Utf8 Ljava/lang/Object;
#22 = Utf8 SourceFile
#23 = Utf8 Test2.java
#24 = NameAndType #10:#11 // "<init>":()V
#25 = Utf8 jvm/classFile/Test2
#26 = Utf8 jvm/classFile/Test1
#27 = Utf8 java/io/Serializable
#28 = Utf8 java/lang/Comparable
{
public jvm.classFile.Test2();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method jvm/classFile/Test1."<init>":()V
4: return
LineNumberTable:
line 10: 0
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Ljvm/classFile/Test2;
public void mathod();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=0, locals=1, args_size=1
0: return
LineNumberTable:
line 18: 0
LocalVariableTable:
Start Length Slot Name Signature
0 1 0 this Ljvm/classFile/Test2;
public int compareTo(java.lang.Object);
descriptor: (Ljava/lang/Object;)I
flags: ACC_PUBLIC
Code:
stack=1, locals=2, args_size=2
0: iconst_0
1: ireturn
LineNumberTable:
line 23: 0
LocalVariableTable:
Start Length Slot Name Signature
0 2 0 this Ljvm/classFile/Test2;
0 2 1 o Ljava/lang/Object;
}
SourceFile: "Test2.java"
解释:
第一步,到常量池里面找#1常量(#1:MethodRef #3,#24)
#3 指向Class_info的索引项(u2类型数据)
#24 指向的就NameAndType的索引项(u2类型数据)
第二步:找#3号常量(#26)
第三步:找#26号常量(#26号常量是utf-8编码的字符串: Test1)
第四步:找#24号常量,NameAndType的数据结构(u2:名称常量项的索引;u2:描述符常量项的索引;#10:#11)
第五步:#10对应<init>
第六步:#11对应()V
搜索的结果:V Test1.<init>()
4.init和clinit
init:实例化初始化方法
调用new初始化对象的时候;
调用反射的时候newInstance();
调用clone方法的时候
ObjectInpustream.getObject序列化的时候;
clinit:类和接口的初始化
所有的类变量初始化语句和静态初始化语句都被Java编译搜集到一起,放到clinit;
访问标识
用于识别一些类或者接口层次的访问信息,包括:这个Class是类还是接口,是否为public,是否为abstract,类是否声明为final等
访问标识之后顺序排列:类索引,父类索引,接口索引集合;
接口索引集合入口第一项是u2类型的接口计数器,表示索引表的容量(实现的接口书),如果该类没有实现任何接口,则计数器的值为0,实现多少个接口,每个接口都需要u2个单位