注意:本章内容紧接上篇文章 通过HelloWorld读懂Java类(.class)文件结构
字节码指令
通过读取HelloWorld.class
文件的内容,我们得知方法信息
部分包含字节码指令。
如:<init>
方法部分的字节码指令为2A B7 00 01 B1
其中:
2A
代表aload_0
,即加载slot 0的局部变量,即this,作为下面invokespecial
的参数
B7
代表invokespecial
,即预备调用构造方法
00 01
代表常量池中的第一项,即Method java/lang/Object."<init>":()V
。
B1
代表return
,即返回。
main
方法部分的字节码指令为B2 00 02 12 03 B6 00 04 B1
其中:
B2
代表getstatic
,即加载静态变量。
00 02
代表常量池中的第二项,即Field java/lang/System.out:Ljava/io/PrintStream;
。
12
代表ldc
,即加载参数
03
代表常量池中的第三项,即String Hello World
B6
代表invokevirtual
,即预备调用成员方法。
00 04
代表常量池中的第四项,即Method java/io/PrintStream.println:(Ljava/lang/String;)V
。
B1
代表return
,即返回。
使用javap工具
通过WinHex工具可以查看字节码文件的内容,但是都是十六进制的,内容比较难懂,查阅也比较繁琐,官方为我们提供了一个工具javap
,可以查看字节码内容的详细信息,且是易于理解的。
D:\java\jdk1.8.0_171\bin>javap -v D:\Workspace\hello-jvm\out\production\hello-jvm\top\zysite\HelloWorld.class
Classfile /D:/Workspace/hello-jvm/out/production/hello-jvm/top/zysite/HelloWorld.class
Last modified 2020-10-27; size 555 bytes
MD5 checksum 067107bd7ba63eb33eb754b67f97dcc0
Compiled from "HelloWorld.java"
public class top.zysite.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 // top/zysite/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 Ltop/zysite/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 top/zysite/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 top.zysite.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 9: 0
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Ltop/zysite/HelloWorld;
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=1, args_size=1
0: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
3: ldc #3 // String Hello World
5: invokevirtual #4 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
8: return
LineNumberTable:
line 11: 0
line 12: 8
LocalVariableTable:
Start Length Slot Name Signature
0 9 0 args [Ljava/lang/String;
}
SourceFile: "HelloWorld.java"