目录
在介绍编译和反编译之前,我们先来简单介绍下编程语言。编程语言分为低级语言和高级语言。
机器语言和汇编语言属于低级语言,直接用计算机指令编写程序。
而C、C++、Java、Python等属于高级语言,用语句编写程序,语句是计算机指令的抽象表示。
上面提到语言有两种,一种低级语言,一种高级语言。简单的理解:低级语言是计算机认识的语言、高级语言是程序员认识的语言。
那么如何从高级语言转换成低级语言呢?这个过程其实就是编译。
将便于人编写、阅读、维护的高级计算机语言所写作的源代码程序,翻译为计算机能解读、运行的低阶机器语言的程序的过程就是编译。负责这一过程的处理的工具叫做编译器。
现在我们知道了什么是编译,也知道了什么是编译器。不同的语言都有自己的编译器,Java语言中负责编译的编译器是一个命令:javac
当我们写完一个 HelloWorld.java 文件后,我们可以使用 javac HelloWorld.java 命令来生成HelloWorld.class文件,这个class类型的文件是JVM可以识别的文件。通常我们认为这个过程叫做Java语言的编译。其实,class文件仍然不是机器能够识别的语言,因为机器只能识别机器语言,还需要JVM再将这种class文件类型字节码转换成机器可以识别的机器语言。
javac是收录于JDK中的Java语言编译器。该工具可以将后缀名为.java的源文件编译为后缀名为.class的可以运行于Java虚拟机的字节码。
反编译的过程与编译刚好相反,就是将已编译好的编程语言还原到未编译的状态,也就是找出程序语言的源代码。就是将机器看得懂的语言转换成程序员可以看得懂的语言。Java语言中的反编译一般指将class文件转换成java文件。
有了反编译工具,我们可以做很多事情,最主要的功能就是有了反编译工具,我们就能读得懂Java编译器生成的字节码。比如我们就可以洞悉Java语法糖背后的原理。
javap并没有将字节码反编译成java文件,而是生成了一种我们可以看得懂字节码。其实javap生成的文件仍然是字节码,只是程序员可以稍微看得懂一些。
拿一段简单的代码举例
- public class switchDemoString {
- public static void main(String[] args) {
- String str = "world";
- switch (str) {
- case "hello":
- System.out.println("hello");
- break;
- case "world":
- System.out.println("world");
- break;
- default:
- break;
- }
- }
- }
执行以下两个命令:
javac Decompilation.java
javap -c Decompilation.class
- public class SwitchDemoString {
- public SwitchDemoString();
- Code:
- 0: aload_0
- 1: invokespecial #1 // Method java/lang/Object."
":()V - 4: return
-
- public static void main(java.lang.String[]);
- Code:
- 0: ldc #2 // String world
- 2: astore_1
- 3: aload_1
- 4: astore_2
- 5: iconst_m1
- 6: istore_3
- 7: aload_2
- 8: invokevirtual #3 // Method java/lang/String.hashCode:()I
- 11: lookupswitch { // 2
- 99162322: 36
- 113318802: 50
- default: 61
- }
- 36: aload_2
- 37: ldc #4 // String hello
- 39: invokevirtual #5 // Method java/lang/String.equals:(Ljava/lang/Object;)Z
- 42: ifeq 61
- 45: iconst_0
- 46: istore_3
- 47: goto 61
- 50: aload_2
- 51: ldc #2 // String world
- 53: invokevirtual #5 // Method java/lang/String.equals:(Ljava/lang/Object;)Z
- 56: ifeq 61
- 59: iconst_1
- 60: istore_3
- 61: iload_3
- 62: lookupswitch { // 2
- 0: 88
- 1: 99
- default: 110
- }
- 88: getstatic #6 // Field java/lang/System.out:Ljava/io/PrintStream;
- 91: ldc #4 // String hello
- 93: invokevirtual #7 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
- 96: goto 110
- 99: getstatic #6 // Field java/lang/System.out:Ljava/io/PrintStream;
- 102: ldc #2 // String world
- 104: invokevirtual #7 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
- 107: goto 110
- 110: return
- }
参考文章: