Java| 编译和反编译

article/2025/8/24 18:46:58

什么是编程语言?

在介绍编译和反编译之前,我们先来简单介绍下编程语言(Programming Language)。编程语言(Programming Language)分为低级语言(Low-level Language)和高级语言(High-level Language)
机器语言(Machine Language)和汇编语言(Assembly Language)属于低级语言,直接用计算机指令编写程序。
而C、C++、Java、Python等属于高级语言,用语句(Statement)编写程序,语句是计算机指令的抽象表示。

什么是编译?

上面提到语言有两种,一种低级语言,一种高级语言。简单的理解:低级语言是计算机认识的语言、高级语言是程序员认识的语言。
那么如何从高级语言转换成低级语言呢?这个过程其实就是编译


将便于人编写、阅读、维护的高级计算机语言所写作的源代码程序,翻译为计算机能解读、运行的低阶机器语言的程序的过程就是编译。负责这一过程的处理的工具叫做编译器


现在我们知道了什么是编译,也知道了什么是编译器。不同的语言都有自己的编译器,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语法糖背后的原理。

Java常用反编译工具

本文主要介绍4个Java的反编译工具:javapjadcfr以及可视化反编译工具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_01: invokespecial #8                  // Method java/lang/Object."<init>":()V4: returnpublic static void main(java.lang.String[]);Code:0: ldc           #16                 // String world2: astore_13: aload_14: dup5: astore_26: invokevirtual #18                 // Method java/lang/String.hashCode:()I9: lookupswitch  { // 299162322: 36113318802: 48default: 82}36: aload_237: ldc           #24                 // String hello39: invokevirtual #26                 // Method java/lang/String.equals:(Ljava/lang/Object;)Z42: ifne          6045: goto          8248: aload_249: ldc           #16                 // String world51: invokevirtual #26                 // Method java/lang/String.equals:(Ljava/lang/Object;)Z54: ifne          7157: goto          8260: getstatic     #30                 // Field java/lang/System.out:Ljava/io/PrintStream;63: ldc           #24                 // String hello65: invokevirtual #36                 // Method java/io/PrintStream.println:(Ljava/lang/String;)V68: goto          8271: getstatic     #30                 // Field java/lang/System.out:Ljava/io/PrintStream;74: ldc           #16                 // String world76: invokevirtual #36                 // Method java/io/PrintStream.println:(Ljava/lang/String;)V79: goto          8282: return
}

javap并没有将字节码反编译成java文件,而是生成了一种我们可以看得懂字节码。其实javap生成的文件仍然是字节码,只是程序员可以稍微看得懂一些。如果你对字节码有所掌握,还是可以看得懂以上的代码的。其实就是把String转成hashcode,然后进行比较。


个人认为,一般情况下我们会用到javap命令的时候不多,一般只有在真的需要看字节码的时候才会用到。但是字节码中间暴露的东西是最全的,你肯定有机会用到,比如我在分析synchronized的原理的时候就有是用到javap。通过javap生成的字节码,我发现synchronized底层依赖了ACC_SYNCHRONIZED标记monitorentermonitorexit两个指令来实现同步。

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()方法来实现的。


PS: 但是,由于JAD已经很久不更新了,在对Java7生成的字节码进行反编译时,偶尔会出现不支持的问题,在对Java 8的lambda表达式反编译时就彻底失败。

CFR

JAD很好用,但是无奈的是很久没更新了,所以只能用一款新的工具替代他,CFR是一个不错的选择,相比JAD来说,他的语法可能会稍微复杂一些,但是好在他可以用.


CFR将反编译现代Java特性–Java 8 lambdas(Java和更早版本中的Java beta 103),已经反编译Java 7 String,但CFR是完全用Java 6编写的.

我们使用CFR对刚刚的代码进行反编译。执行一下命令:
java -jar cfr_0_125.jar Decompilation.class --decodestringswitch false
得到以下错误的结果(死活是反编译失败~)

/** Decompiled with CFR 0_125.*/
package com.yveshe;public class Decompilation {/** Exception decompiling*/public static void main(String[] args) {// This method has failed to decompile.  When submitting a bug report, please provide this stack trace, and (if you hold appropriate legal rights) the relevant class file.// org.benf.cfr.reader.util.CannotPerformDecode: reachable test BLOCK was exited and re-entered.// org.benf.cfr.reader.bytecode.analysis.opgraph.op3rewriters.Misc.getFarthestReachableInRange(Misc.java:143)// org.benf.cfr.reader.bytecode.analysis.opgraph.op3rewriters.SwitchReplacer.examineSwitchContiguity(SwitchReplacer.java:385)// org.benf.cfr.reader.bytecode.analysis.opgraph.op3rewriters.SwitchReplacer.replaceRawSwitches(SwitchReplacer.java:65)// org.benf.cfr.reader.bytecode.CodeAnalyser.getAnalysisInner(CodeAnalyser.java:394)// org.benf.cfr.reader.bytecode.CodeAnalyser.getAnalysisOrWrapFail(CodeAnalyser.java:191)// org.benf.cfr.reader.bytecode.CodeAnalyser.getAnalysis(CodeAnalyser.java:136)// org.benf.cfr.reader.entities.attributes.AttributeCode.analyse(AttributeCode.java:95)// org.benf.cfr.reader.entities.Method.analyse(Method.java:369)// org.benf.cfr.reader.entities.ClassFile.analyseMid(ClassFile.java:770)// org.benf.cfr.reader.entities.ClassFile.analyseTop(ClassFile.java:702)// org.benf.cfr.reader.Main.doClass(Main.java:46)// org.benf.cfr.reader.Main.main(Main.java:191)throw new IllegalStateException("Decompilation failed");}
}

中间出了一个小插曲,就是始终反编译失败,由于博主偷懒每次都是用的Eclipse的自动编译生成的class文件,没有通过javac命令你来生成class文件,导致之前存在问题的java文件编译成了class文件而一直么有更新…害死人啊,建议大家手动通过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表达式进行反编译。


CFR还有很多其他参数,均用于不同场景,读者可以使用java -jar cfr_0_125.jar --help进行了解。这里不逐一介绍了。

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不是默认设置,设置成默认设置既可.

PS: 该方式好像不能正常工作了(2019/4/2)更新,可以使用安装jd插件jd-eclipse-site-1.0.0-RC2.zip下载地址 http://jd.benow.ca/ ,参考Eclipse中安装SVN插件(离线安装)方式安装.

如何防止反编译?

由于我们有工具可以对Class文件进行反编译,所以,对开发人员来说,如何保护Java程序就变成了一个非常重要的挑战。但是,魔高一尺、道高一丈。当然有对应的技术可以应对反编译咯。但是,这里还是要说明一点,和网络安全的防护一样,无论做出多少努力,其实都只是提高攻击者的成本而已。无法彻底防治。
典型的应对策略有以下几种:
● 隔离Java程序
○ 让用户接触不到你的Class文件
● 对Class文件进行加密
○ 提到破解难度
● 代码混淆
○ 将代码转换成功能上等价,但是难于阅读和理解的形式


比如: 用很复杂的算法加密 class文件,然后在虚拟机载入前调用解密程序。考虑使用jvmti,这样可以防止class loader被反编译导致加解密算法泄漏.

相关资源

在线反编译:
http://www.javadecompilers.com/ (支持选择多种反编译器)
http://javare.cn/


资源下载:
https://varaneckas.com/jad/ (JAD支持各种平台)
http://jd.benow.ca/ (JD相关)
https://baike.xsoftlab.net/view/264.html (JD-GUI)
http://www.benf.org/other/cfr/ (CFR)
http://jadclipse.sourceforge.net/wiki/index.php/Main_Page (JadClipse:Eclipse插件,也可以通过配置外部的Jad来在Eclipse中实现反编译)


反编译软件(JAD,JadClipse,JD-GUI,CRF)打包下载


参考链接:
http://www.admin10000.com/document/5064.html (7款开源Java反编译工具:)
https://blog.csdn.net/chenchunlin526/article/details/78259682 (反编译工具对比)
http://53873039oycg.iteye.com/blog/2015192(工具CFR,Procyon简介:)


http://chatgpt.dhexx.cn/article/7uVXGDQ8.shtml

相关文章

安装java编译器

安装JDK。 参考&#xff1a;https://www.cnblogs.com/mr-wuxiansheng/p/6850437.html 1.官网下载 Java SE Development Kit 13.0.1 &#xff08;由于是访问国外网站&#xff0c;所以会比较慢。&#xff09; 最好下载EXE版本的&#xff0c;这样什么都不用管&#xff0c;点安装…

java编译和运行

java应用程序的基本结构编写源文件保存源文件额外附加编译器&#xff08;javac.exe&#xff09;解释器&#xff08;java.exe&#xff09; 总结&#xff1a; 假如我的B.java源文件在C:\Users\AUSU\Desktop\ts里面 一般都是进入到这个目录里面编译解释 编译&#xff1a;javac …

java如何编译运行?

对于一个Java开发者来说我们编写的程序肯定是要运行才能体现出作用来&#xff0c;对于新手同学来说你知道如何去编译和运行一个Java程序吗&#xff1f;小千今天就来告诉大家&#xff0c;步骤很详细带好小本本哦。 Java程序编译运行步骤 1.首先我们在windows环境下需要安装好Jav…

JVM(一)一文读懂Java编译全过程

一文读懂Java编译全过程 java代码首先要通过前端编译器编译成.class字节码文件&#xff0c;然后再按一定的规则加载到JVM&#xff08;java 虚拟机&#xff09;内运行&#xff0c;有三种运行方式&#xff0c;解释模式&#xff08;javac&#xff09;、编译模式&#xff08;C1 JIT…

无需插件修改chrome浏览器UA标识为手机版

可能在手机上我们可以很方便的操作将网页的UA设置为电脑版或者是其他的&#xff0c;但是在电脑上一般没有直接的操作修改&#xff0c;网上的经验也一般为使用User-Agent Switcher 插件,修改。其实使用开发者工具就已经能够很方便的修改UA了 首先在对应的页面按下F12&#xff0c…

使用PC端谷歌Chrome浏览器浏览手机网页 修改PC端谷歌Chrome浏览器UA

废话不多说直接上干货&#xff08;快下班了&#xff09;&#xff1a; 1.进入你想要审查元素的网站&#xff0c;比如www.baidu.com&#xff1b; 2.按F12召唤开发者窗口&#xff1b; 3.点击下图框出来的三个点&#xff1a; 4.点击下图框出来的选项&#xff1a; 5.将下面框选住…

【测试】抓包获取浏览器UA,并使用Chrome 调试工具模拟手机

抓包获取浏览器UA 首先需要先安装charles&#xff0c;在http的请求头当中可以看到User-Agent&#xff0c;复制该User-Agent的内容 如果你是前端开发&#xff0c;也可通过以下代码获取UA $(‘body’).html(navigator.userAgent);构造UA 然后打开chrome浏览器的调试工具&…

自定义浏览器UA标识

正文 手机浏览器改成下面这段&#xff0c;可以享受到百度的纯净浏览。 Mozilla/5.0 (Linux; U; Android 10; zh-CN; 2014811 Build/QQ3A.200805.001) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/78.0.2564.116 Quark/3.8.2.126 Mobile Safari/537.36 T7/10.…

修改浏览器UA的作用汇总

1.浏览器标识&#xff08;UA&#xff09; 可以使得服务器能够识别客户使用的操作系统及版本、CPU 类型、浏览器及版本、浏览器渲染引擎、浏览器语言、浏览器插件&#xff0c;从而判断用户是使用电脑浏览还是手机浏览&#xff0c;让网页作出自动的适应。    可理解为网站通过…

一篇文章带你了解清楚什么是UA

最近在做一部分的UA的统计、正好做下记录、有需要的同学也可以借鉴一下 一、首先讲一下UA是什么 UA&#xff0c;全称为“User Agent”&#xff0c;中文通常直译为“用户代理”&#xff0c;但从其产生作用的机制来看&#xff0c;或许称之为“浏览器标识”或“浏览器名片”更为妥…

小白想学大数据?

一.大数据领域的从业人员,应该牢牢把握2018年大数据这三个大的技术方向: 1、Hadoop大数据开发方向; 2、数据挖掘、数据分析和机器学习方向; 3、大数据运维和云计算方向。精通任何方向之一者,均会前(钱)途无量。 3个方向中,大数据开发是基础。以Hadoop开发工程师为例,…

Java开发者,我到底要不要学大数据开发?

一入编程深似海&#xff0c;从此女神是路人。没办法&#xff0c;这行就这样。你不学Spring&#xff0c;总不是跑去学JVM/微服务架构/分布式去了&#xff0c;不断学习根本避免不了。所以关键在于把时间投在学什么上比较划算。 明确表达我的观点&#xff0c;作为一名Javaer进阶大…

从0开始学大数据-数据仓库建模

为什么要数据仓库建模 数据模型是数据组织和存储方法&#xff0c;它强调从业务、数据存取和使用角度合理存储数据。有了适合业务和基础数据存储环境的模型&#xff0c;那么大数据就能获得以下好处&#xff1a; 性能&#xff1a;良好的数据模型能帮助我们快速查询所需要的数据&a…

【金猿案例展】某国家级研究所——组学大数据分析平台建设

荣联科技集团案例 本案例由荣联科技集团投递并参与“数据猿年度金猿策划活动——2020大数据产业创新服务企业榜单及奖项”评选。 大数据产业创新服务媒体 ——聚焦数据 改变商业 大数据时代下&#xff0c;科学大数据已经成为科技创新和社会经济发展的新动力。生物信息学经过近…

零基础学大数据分析现实吗

学习大数据分析已然成为社会的大势所趋&#xff0c;随着传统公司的被迫更新&#xff0c;新兴公司的数据人才增多&#xff0c;大数据技术显得格外的重要。当大数据的大浪凶猛袭来时&#xff0c;要么你冲上浪尖&#xff0c;做时代的弄潮儿&#xff0c;要么被打入海底&#xff0c;…

零基础可以学大数据分析吗

国家的大数据战略实施已经到了关键的落地时期&#xff0c;创新、实体经济都跟大数据分析有着深度的融合&#xff0c;在各行各业都能看到大数据分析师的身影&#xff0c;另外大数据安全管理和立法等方面也进入到了关键时刻&#xff0c;所有这些无不在向人们诉说着一个事实&#…

学大数据需要具备四种条件?你具备几种?

现在学大数据的越来越多了&#xff0c;可是你知道学大数据需要什么条件吗&#xff1f;我来说说四个必备条件。 大数据现已成为年代开展的趋势&#xff0c;很多人纷纷挑选学大数据&#xff0c;想要进入大数据职业。大数据技术体系巨大&#xff0c;包含的常识较多&#xff0c;体…

学大数据专业后悔死了?学大数据以后好就业吗?

前言 从近几年来看&#xff0c;大数据专业的就业情况和就业前景是非常不错的&#xff0c;而所谓的学大数据专业后悔死了等等言论都是不实或偏激的。小编在收集整理相关资料后发现大多数同学认为学大数据专业后悔死了有两个原因&#xff1a;一是大数据专业学习难度大&#xff0…

女生适合学大数据开发吗,女生怎样学大数据开发

红遍全球的大数据开发,不仅仅吸引了男孩子的目光,更是让不少女孩子为之兴奋。但是有些女孩子被刻板成见所击败,不敢涉足大数据开发。那么,女生适合学大数据开发吗?女生应该怎样学习大数据开发? 女生适合学大数据开发吗: 1、女生适不适合学大数据开发这个问题,就跟女生…

哪些人适合学大数据分析

哪些人适合学大数据分析&#xff0c;实际上&#xff0c;问题还有个潜台词是“什么人学习数据分析&#xff0c;会更容易取得成功(比如职业成功)”&#xff0c;这个要关乎你的兴趣、付出和机遇。但要做到出类拔萃&#xff0c;除了上面三点&#xff0c;还需要一点天赋&#xff0c;…