分析的条件: 基于上一篇的16进制字节码及反编译结果(javap -verbose the desire class)
分析的时候可以使用(jclasslib)进行查看,也提供了IDEA的plugin,显示的信息比较完善,比较优雅的分门别类字节码的相关信息。
分析的结果
反编译结果:
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的数据类型
网友评论