您好,欢迎访问一九零五行业门户网

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.javajavap -c decompilation.class
生成代码如下:
compiled from "decompilation.java"public class decompilation { public decompilation(); code: 0: aload_0 1: invokespecial #8 // method java/lang/object."<init>":()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.javapackage 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主要的是提供了可视化操作,直接拖拽文件到窗口既可,效果图如下
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:
基本配置完毕后,我们可以设置一下class文件的默认打开方式:
eclipse->window->preferences->general->editors->file associations 我们可以看到class文件的打开方式有两个,这里设置jadclipse和eclipse自带的class file viewer,而jadclipse是默认的。
全部配置完成,下面我们可以查看源码了,选择需要查看的类,按f3即可查看源码.如果jadclipse不是默认设置,设置成默认设置既可.
更多java知识请关注java基础教程栏目。
以上就是java如何进行反编译的详细内容。
其它类似信息

推荐信息