在初学 java 的时候,我们都学习过 javac 和 java 在编译和运行中的作用,关于 javap 的了解不甚了了。javap 的目的是为了帮助开发者深入了解 java 编译器的机制。下面我将会用具体实例展示过程,和大家分享一下javap的内容。
源代码
public class test {
private static int classV =2;
public static void main(String[] args) {
classV =200;
int localV =4;
localV =400;
}
}
二进制
idea bin_ed插件查看。
看不懂 那就使用人能看的懂的汇编语言查看类文件结构和代码指令。
javap 指令和选项
0:无选项
打印package, protected and public fields, and methods
public class com.example.test {
public com.example.test();
public static void main(java.lang.String[]);
static {};
}
1:辅助指令
-help
–help
-?
2:代码行号和方法的局部变量表
-l
public class com.example.test {
//默认构造方法
public com.example.test();
//代码行号:命令偏移位置
LineNumberTable:
line 3: 0
//局部变量表
LocalVariableTable:
Start Length Slot Name Signature
0 5 0 this Lcom/example/test;
public static void main(java.lang.String[]);
//代码行号:命令偏移位置
LineNumberTable:
line 6: 0
line 7: 6
line 8: 8
line 9: 12
//局部变量表
LocalVariableTable:
Start Length Slot Name Signature
0 13 0 args [Ljava/lang/String; //方法参数
8 5 1 localV I。/局部变量localV
//静态代码块
static {};
LineNumberTable:
line 4: 0
}
3 用级别过滤方法 属性 类
-public
-protected
-private
-p
4.反汇编 出汇编指令
javap -c
//纯汇编指令
public class com.example.test {
public com.example.test();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
public static void main(java.lang.String[]);
Code:
0: sipush 200
3: putstatic #2 // Field classV:I
6: iconst_4
7: istore_1
8: sipush 400
11: istore_1
12: return
static {};
Code:
0: iconst_2
1: putstatic #2 // Field classV:I
4: return
}
5 显示verbose详细信息
javap -v
Classfile /Users/zhangshanxue/Downloads/akka-quickstart-java/target/classes/com/example/test.class
//javap -sysinfo显示下面3行
Last modified 2021-4-5; size 507 bytes
MD5 checksum 24a0c74751aafd61d0f7f69be9c161db
Compiled from "test.java"
public class com.example.test
//即1.8 对照表和原因见附录1
//u2类型 即每个占用两个字节
minor version: 0
major version: 52
//类标志 见附录2
//u2类型access_flags 通过位与表示多个权限
flags: ACC_PUBLIC, ACC_SUPER
//常量池。class文件结构重点
//u2 2两个字节 表示数量
//pool 常量池
//constant_pool_count 和constant_pool[]表示为常量池中内容
//后面大部分内容依赖此常量池
Constant pool:
#1 = Methodref #4.#22 // java/lang/Object."<init>":()V
#2 = Fieldref #3.#23 // com/example/test.classV:I
#3 = Class #24 // com/example/test
#4 = Class #25 // java/lang/Object
#5 = Utf8 classV
#6 = Utf8 I
#7 = Utf8 <init>
#8 = Utf8 ()V
#9 = Utf8 Code
#10 = Utf8 LineNumberTable
#11 = Utf8 LocalVariableTable
#12 = Utf8 this
#13 = Utf8 Lcom/example/test;
#14 = Utf8 main
#15 = Utf8 ([Ljava/lang/String;)V
#16 = Utf8 args
#17 = Utf8 [Ljava/lang/String;
#18 = Utf8 localV
#19 = Utf8 <clinit>
#20 = Utf8 SourceFile
#21 = Utf8 test.java
#22 = NameAndType #7:#8 // "<init>":()V
#23 = NameAndType #5:#6 // classV:I
#24 = Utf8 com/example/test
#25 = Utf8 java/lang/Object
{
public com.example.test();
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
//Javap -l显示下面信息
//行号和上面对应的指令偏移位置
LineNumberTable:
line 3: 0
LocalVariableTable:
//局部变量表 在指令偏移位置start start+length之间有效
Start Length Slot Name Signature
0 5 0 this Lcom/example/test;
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
//代码汇编指令
stack=1, locals=2, args_size=1
0: sipush 200
3: putstatic #2 // Field classV:I
6: iconst_4
7: istore_1
8: sipush 400
11: istore_1
12: return
//Javap -l显示下面信息
//行号和上面对应的指令偏移位置
LineNumberTable:
line 6: 0
line 7: 6
line 8: 8
line 9: 12
LocalVariableTable:
Start Length Slot Name Signature
0 13 0 args [Ljava/lang/String;
8 5 1 localV I
static {};
descriptor: ()V
flags: ACC_STATIC
Code:
stack=1, locals=0, args_size=0
0: iconst_2
1: putstatic #2 // Field classV:I
4: return
//行号和上面对应的指令偏移位置
LineNumberTable:
line 4: 0
附录1
1.1从45开始因为正式发布之前可能 其他版本号可能已经被用了
1.5改为5.0 也是差不多原因 为了表示重要性更名
Corresponding major version 指定版本 和"Supported major versions"兼容范围
附录2
附录3
直接分析字节码块
1为了方便交流表达class文件的结构
使用u1 u2 u4 u8无符号数表示字节数使用*_info 结尾表示池(数组cp_info、field_info、method_info、attribute_info
ClassFile {
//表示java class的文件格式 固定为cafe baby 4个字节
u4 magic;
//主版本号和次版本号共同决定了类文件格式的版本
//u2类型 即每个占用两个字节
//56包含56以后support for N.0 and N.65535
u2 minor_version;
//主版本号56(java12)之前 minjor只支持0
u2 major_version;
//常量池数量
u2 constant_pool_count;
cp_info constant_pool[constant_pool_count-1];
u2 access_flags;
//this_class、super_class、interfaces指向常量池的CONSTANT_Class_info
u2 this_class;
u2 super_class;
u2 interfaces_count;
u2 interfaces[interfaces_count];
u2 fields_count;
//指向常量池structures:CONSTANT_Fieldref_info 结构
field_info fields[fields_count];
u2 methods_count;
//指向常量池CONSTANT_Methodref_info 结构
method_info methods[methods_count];
u2 attributes_count;
attribute_info attributes[attributes_count];
}
附录4
常量池字节码结构和常量池结构部分对应关系
以上就是关于 java 底层知识的拆解字节码文件 javap 命令的详细内容,想要了解更多关于javap命令和java底层知识的其他资料,请关注W3Cschool其它相关文章,也希望大家能够多多地支持!