java怎么反编译_java如何进行反编译

java怎么反编译_java如何进行反编译反编译的过程与编译刚好相反,就是将已编译好的编程语言还原到未编译的状态,也就是找出程序语言的源代码。就是将机器看得懂的语言转换成程序员可以看得懂的语言。Java语言中的反编译一般指将class文件转换成java文件。Java常用反编译工具本文主要介绍4个Java的反编译工具:javap、jad和cfr以及可视化反编译工具JD-GUIJAVAPjavap是jdk自带的一个工具,可以对代码反编译,也可…

大家好,欢迎来到IT知识分享网。

d85fc7965f2b7780d1b832a9ed6c7654.png

反编译的过程与编译刚好相反,就是将已编译好的编程语言还原到未编译的状态,也就是找出程序语言的源代码。就是将机器看得懂的语言转换成程序员可以看得懂的语言。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反编译后内容如下:

命令:jad.exe Decompilation.class 会生成一个Decompilation.jad的文件

JAD反编译的结果如下:// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.

// Jad home page: http://www.kpdus.com/jad.html

// 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: {

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主要的是提供了可视化操作,直接拖拽文件到窗口既可,效果图如下

ab7834df99ef504ce812b6e254f91ca3.png

JadClipse

在Eclipse中安装Jad插件,注意这里是安装的是Jad插件不是Jd插件~

所需要资源: net.sf.jadclipse_3.3.0.jar插件jar和JAD.exe反编译软件(在文末有下载地址)

JadClipse下载地址在官网下载插件的jar包,然后将jar包放到eclipse的plugins目录下;在打开Eclipse,Eclipse->Window->Preferences->Java,此时你会发现会比原来多了一个JadClipse的选项如下图配置JadClipse:

4511a765caa09a51b16a5a05ab852dde.png

3a47f36cd6ca9f4dd78590bd2634156d.png

基本配置完毕后,我们可以设置一下class文件的默认打开方式:

Eclipse->Window->Preferences->General->Editors->File Associations 我们可以看到class文件的打开方式有两个,这里设置JadClipse和Eclipse自带的Class File Viewer,而JadClipse是默认的。

全部配置完成,下面我们可以查看源码了,选择需要查看的类,按F3即可查看源码.如果JadClipse不是默认设置,设置成默认设置既可.

更多java知识请关注java基础教程栏目。

免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://yundeesoft.com/24050.html

(0)
上一篇 2023-08-31 13:00
下一篇 2023-08-31 17:00

相关推荐

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注微信