java怎么反编译_java如何进⾏反编译
反编译的过程与编译刚好相反,就是将已编译好的编程语⾔还原到未编译的状态,也就是出程序语⾔的源代码。就是将机器看得懂的语⾔转换成程序员可以看得懂的语⾔。Java语⾔中的反编译⼀般指将class⽂件转换成java⽂件。
Java常⽤反编译⼯具
本⽂主要介绍4个Java的反编译⼯具:javap、jad和cfr以及可视化反编译⼯具JD-GUI
JAVAP
javap是jdk⾃带的⼀个⼯具,可以对代码反编译,也可以查看java编译器⽣成的字节码。javap和其他两个反编译⼯具最⼤的区别是他⽣成的⽂件并不是java⽂件,也不像其他两个⼯具⽣成代码那样更容易理解。拿⼀段简单的代码举例,如我们想分析Java 7中的switch是如何⽀持String的,我们先有以下可以编译通过的源代码: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
⽣成代码如下:Compiled from "Decompilation.java"
public class Decompilation {
public Decompilation();
Code:
0: aload_0
1: invokespecial #8 // Method java/lang/Object."":()V
4: return
public static void main(java.lang.String[]);
Code:
0: ldc #16 // String world
2: astore_1
3: aload_1
4: dup
5: astore_2
6: invokevirtual #18 // Method java/lang/String.hashCode:()I
9: lookupswitch { // 2
99162322: 36
113318802: 48
default: 82
}
36: aload_2
37: ldc #24 // String hello
39: invokevirtual #26 // Method java/lang/String.equals:(Ljava/lang/Object;)Z 42: ifne 60
45: goto 82
48: aload_2
49: ldc #16 // String world
51: invokevirtual #26 // Method java/lang/String.equals:(Ljava/lang/Object;)Z 54: ifne 71
57: goto 82
60: getstatic #30 // Field java/lang/System.out:Ljava/io/PrintStream;
63: ldc #24 // String hello
65: invokevirtual #36 // Method java/io/PrintStream.println:(Ljava/lang/String;)V 68: goto 82
71: getstatic #30 // Field java/lang/System.out:Ljava/io/PrintStream;
74: ldc #16 // String world
76: invokevirtual #36 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
79: goto 82
82: return
}
javap并没有将字节码反编译成java⽂件,⽽是⽣成了⼀种我们可以看得懂字节码。其实javap⽣成的⽂件仍然是字节码,只是程序员可以稍微看得懂⼀些。如果你对字节码有所掌握,还是可以看得懂以上的代码的。其实就是把String转成hashcode,然后进⾏⽐较。
JAD
JAD是⼀个⽐较不错的反编译⼯具,只要下载⼀个执⾏⼯具,就可以实现对class⽂件的反编译了。还是上⾯的源代码,使⽤jad反编译后内容如下:
命令: Decompilation.class 会⽣成⼀个Decompilation.jad的⽂件
JAD反编译的结果如下:// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
// Decompiler options: packimports(3)
// Source File Name: Decompilation.java
package com.yveshe;
import java.io.PrintStream;
public class Decompilation
{
public Decompilation()
{
}
public static void main(String args[])
{
String str = "world";
String s;
switch((s = str).hashCode())
{
default:
break;
case 99162322:
if(s.equals("hello"))
System.out.println("hello");
break;
case 113318802:
if(s.equals("world"))
System.out.println("world");
break;
}
}
}
看上⾯的代码这不就是标准的java的源代码么。这个就很清楚的可以看到原来字符串的switch是通过equals()和hashCode()⽅法来实现的。
CFR
JAD很好⽤,但是⽆奈的是很久没更新了,所以只能⽤⼀款新的⼯具替代他,CFR是⼀个不错的选择,相⽐JAD来说,他的语法可能会稍微复杂⼀些,但是好在他可以⽤.
CFR将反编译现代Java特性–Java 8 lambdas(Java和更早版本中的Java beta 103),已经反编译Java 7 String,但CFR是完全⽤Java 6编写的.
建议⼤家⼿动通过javac Decompilation.java命令来编译⽣成Decompilation.class⽂件,再做测试.
成功的反编译结果如下:/*
* Decompiled with CFR 0_125.
*/
package com.yveshe;
import java.io.PrintStream;
public class Decompilation {
public static void main(String[] args) {
String str;
String s = str = "world";
switch (s.hashCode()) {
default: {
break;
}
case 99162322: {
if (!s.equals("hello")) break;
System.out.println("hello");
break;
}
case 113318802: {
if (!s.equals("world")) break;
System.out.println("world");
}
}
}
}
相⽐Jad来说,CFR有很多参数,还是刚刚的代码,如果我们使⽤以下命令,输出结果就会不同:
E:\CRF>java -jar cfr_0_125.jar Decompilation.class/*
* Decompiled with CFR 0_125.
*/
package com.yveshe;
import java.io.PrintStream;
public class Decompilation {
public static void main(String[] args) {
String str;
String s = str = "world";
switch (s.hashCode()) {
default: {
break;
}
case 99162322: {
if (!s.equals("hello")) break;
System.out.println("hello");
break;
}
case 113318802: {eclipse怎么打开已有的java文件
if (!s.equals("world")) break;
System.out.println("world");
}
}
}
}
--decodestringswitch表⽰对于switch⽀持string的细节进⾏解码。
类似的还有--decodeenumswitch、--decodefinally、--decodelambdas等。
--decodelambdas可以对lambda表达式进⾏反编译。
JD-GUI
JD-GUI 是⼀个⽤ C++ 开发的 Java反编译⼯具,由 Pavel Kouznetsov开发,⽀持Windows、Linux和
苹果Mac Os三个平台。⽽且提供了Eclipse平台下的插件JD-Eclipse。JD-GUI 基于GPLv3开源协议,对个⼈使⽤是完全免费的。JD-GUI主要的是提供了可视化操作,直接拖拽⽂件到窗⼝既可,效果图如下

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。