共计 2482 个字符,预计需要花费 7 分钟才能阅读完成。
编译和反编译
编程语言分成高级语言和低级语言。低级语言如机器语言、汇编语言。这类语言间接用计算机指令编写命令,不须要编译。这些语言机器能看到懂,然而程序员读起来很吃力。而咱们平时常常用的语言 C、Java、Python 属于高级语言,这些语言程序员能看的懂。而机器是看不懂的。
简略的总结为:高级语言就是程序员意识的语言,而低级语言是机器意识的语言。而把高级语言转成低级语言这个过程就是编译,而反编译就是把低级语言转成高级语言。
有了反编译,咱们就可以看懂 Java 编译器生成的字节码,比方 Synchronized 的实现原理(监听器 monitor)、枚举、语法糖、泛型,这些都须要用到反编译工具。
javap
javap 是 jdk 自带的反编译命令,能够对代码进行反编译,然而反编译的并不是 java 文件。
应用格局
javap <options> <classes>
罕用: javap -c 类名
-help --help -? 输入此用法音讯
-version 版本信息
-v -verbose 输入附加信息
-l 输入行号和本地变量表
-public 仅显示公共类和成员
-protected 显示受爱护的 / 公共类和成员
-package 显示程序包 / 受爱护的 / 公共类和成员 (默认)
-p -private 显示所有类和成员
-c 对代码进行反汇编
-s 输入外部类型签名
-sysinfo 显示正在解决的类的零碎信息 (门路, 大小, 日期, MD5 散列)
-constants 显示最终常量
-classpath <path> 指定查找用户类文件的地位
-cp <path> 指定查找用户类文件的地位
-bootclasspath <path> 笼罩疏导类文件的地位
上面写一段 synchronized 代码:
public class SynchronizedTest {
private int count = 0;
public void addOne() {synchronized (SynchronizedTest.class) {count++;}
}
}
执行编译和反编译命令
javac SynchronizedTest .java
javap -c SynchronizedTest.class
间接用记事本关上 SynchronizedTest.class 文件是一堆乱码文件,用 sublime 关上是一串数字
cafe babe 0000 0034 0017 0a00 0400 1209
0003 0013 0700 1407 0015 0100 0563 6f75
6e74 0100 0149 0100 063c 696e 6974 3e01
0003 2829 5601 0004 436f 6465 0100 0f4c
696e 654e 756d 6265 7254 6162 6c65 0100
0661 6464 4f6e 6501 000d 5374 6163 6b4d
6170 5461 626c 6507 0014 0700 1507 0016
反编译后的代码:
public class com.SynchronizedTest {public com.SynchronizedTest();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: aload_0
5: iconst_0
6: putfield #2 // Field count:I
9: return
public void addOne();
Code:
0: ldc #3 // class com/yyw/oil/web/admin/controller/purchase/SynchronizedTest
2: dup
3: astore_1
4: monitorenter
5: aload_0
6: dup
7: getfield #2 // Field count:I
10: iconst_1
11: iadd
12: putfield #2 // Field count:I
15: aload_1
16: monitorexit
17: goto 25
20: astore_2
21: aload_1
22: monitorexit
23: aload_2
24: athrow
25: return
Exception table:
from to target type
5 17 20 any
20 23 20 any
}
javap 并没有将字节码反编译成成 java 文件,而是生成一种另一种能看的懂得字节码。能够看出被 synchronized 润饰的代码蕴含 monitorenter 和 monitorexit。synchronized 底层依赖着两个指令来实现同步,这里看起来比拟艰涩难懂。
CFR
在官网上下载 jar,执行如下命令:
java -jar cfr-0.151.jar SynchronizedTest.class
失去反编译 java 文件:
public class SynchronizedTest {
private int count = 0;
/*
* WARNING - Removed try catching itself - possible behaviour change.
*/
public void addOne() {
Class<SynchronizedTest> clazz = SynchronizedTest.class;
synchronized (SynchronizedTest.class) {
++this.count;
// ** MonitorExit[var1_1] (shouldn't be in output)
return;
}
}
}
CFR 还带有一些参数:
参数 | 正文 |
---|---|
–decodeenumswitch (boolean) | 去除 switch 对枚举反对的语法糖 |
–decodelambdas (boolean) | 去除 lambda 表达式的语法糖 |
–decodestringswitch (boolean) | 去除 switch string 反对的语法糖 |
其余参数可应用如下命令查看:
java -jar cfr-0.151.jar --help
idea
应用 idea 生成 class 文件,用 idea 关上 class 文件即可。idea 是绝大多数 Java 程序员应用的编辑器,应用 idea 关上文件比拟不便、快捷。
参考
Java 代码的编译与反编译那些事儿
javap 的应用