要想使得java程序能够在JVM上成功执行,首先将 .java 文件编译为 .class 文件,然后由JVM执行。
.class 文件主要包含以下信息
- 类的基本信息
- 常量池
- 类方法的定义
- JVM指令
下面由一个最简单的java代码,来分析class文件的内容
public class HelloWorld {
public static void main(String[] args) {
System.out.println("hello world");
}
}
执行 javap -v HelloWorld.class 命令,将class文件进行反编译。
结果如下:
第一部分:类的基本信息
包含 类文件的大小,类文件名,最后修改时间,类名,类的访问修饰符。
Classfile /C:/Users/Administrator/Desktop/代码/jvm/out/production/jvm/cn/itcast/jvm/t5/HelloWorld.class
Last modified 2021-11-9; size 567 bytes
MD5 checksum 8efebdac91aa496515fa1c161184e354
Compiled from "HelloWorld.java"
public class cn.itcast.jvm.t5.HelloWorld
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
第二部分:常量池信息,在指令执行时,指令在常量池中查找用到的常量信息,使得指令能够正常执行。
Constant pool:
#1 = Methodref #6.#20 // java/lang/Object."<init>":()V
#2 = Fieldref #21.#22 // java/lang/System.out:Ljava/io/PrintStream;
#3 = String #23 // hello world
#4 = Methodref #24.#25 // java/io/PrintStream.println:(Ljava/lang/String;)V
#5 = Class #26 // cn/itcast/jvm/t5/HelloWorld
#6 = Class #27 // java/lang/Object
#7 = Utf8 <init>
#8 = Utf8 ()V
#9 = Utf8 Code
#10 = Utf8 LineNumberTable
#11 = Utf8 LocalVariableTable
#12 = Utf8 this
#13 = Utf8 Lcn/itcast/jvm/t5/HelloWorld;
#14 = Utf8 main
#15 = Utf8 ([Ljava/lang/String;)V
#16 = Utf8 args
#17 = Utf8 [Ljava/lang/String;
#18 = Utf8 SourceFile
#19 = Utf8 HelloWorld.java
#20 = NameAndType #7:#8 // "<init>":()V
#21 = Class #28 // java/lang/System
#22 = NameAndType #29:#30 // out:Ljava/io/PrintStream;
#23 = Utf8 hello world
#24 = Class #31 // java/io/PrintStream
#25 = NameAndType #32:#33 // println:(Ljava/lang/String;)V
#26 = Utf8 cn/itcast/jvm/t5/HelloWorld
#27 = Utf8 java/lang/Object
#28 = Utf8 java/lang/System
#29 = Utf8 out
#30 = Utf8 Ljava/io/PrintStream;
#31 = Utf8 java/io/PrintStream
#32 = Utf8 println
#33 = Utf8 (Ljava/lang/String;)V
第三部分:类方法的定义
{
public cn.itcast.jvm.t5.HelloWorld(); // 默认的构造方法
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 4: 0
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Lcn/itcast/jvm/t5/HelloWorld;
public static void main(java.lang.String[]); // main方法
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=1, args_size=1
0: getstatic #2 // 根据代码生成的指令
3: ldc #3
5: invokevirtual #4
8: return
LineNumberTable:
line 6: 0
line 7: 8
LocalVariableTable:
Start Length Slot Name Signature
0 9 0 args [Ljava/lang/String;
}
下面是 根据代码 System.out.println(“hello world”); 生成的JVM指令。
来分析以下执行的过程。
0: getstatic #2
3: ldc #3
5: invokevirtual #4
0: getstatic #2
getstatic 是得到一个静态量 #2 是指这个静态量在常量池中的地址。
常量池中的#2 为 :
#2 = Fieldref #21.#22
#2 是一个属性的引用,在常量池中的 #21与#22,对应如下内容
#21 = Class #28
#22 = NameAndType #29:#30
#21 是一个类名,对应在#28 内容如下
#28 = Utf8 java/lang/System
#22 是名字和类型的信息,对应在 #29和#30行
#29 = Utf8 out
#30 = Utf8 Ljava/io/PrintStream;
终于找到了 正确的值,至此结束,向上返回,最终 找到的内容为 java/lang/System.out:Ljava/io/PrintStream;
也就是System.out 的方法
3: ldc #3
ldc指令是加载一个引用地址。
#3 = String #23
#23 = Utf8 hello world
可以看出 最终找到的是 hello world
5: invokevirtual #4
invokevirtual 是执行一次方法调用,通过 #4来找到要执行的方法
#4 = Methodref #24.#25
#24 = Class #31
#25 = NameAndType #32:#33
#31 = Utf8 java/io/PrintStream
#32 = Utf8 println
#33 = Utf8 (Ljava/lang/String;)V
最终要执行的方法是
java/io/PrintStream.println:(Ljava/lang/String;)V
方法的参数是 String类型。
以上的三行JVM指令就对应着 System.out.println(“hello world”); 这个java代码。
常量池,就是一张表,虚拟机指令根据这张常量表找到要执行的类名、方法名、参数类型、字面量
等信息。
当类被加载到内存中,常量池的信息就会放入运行时常量池,并把里面的符号地址变为真实地址。