jad反编译工具,已经不再更新,且只支持JDK1.4,但并不影响其强大的功能。

基本用法:jad xxx.class,会生成直接可读的xxx.jad文件。

自动拆装箱

对于基本类型和包装类型之间的转换,通过xxxValue()和valueOf()两个方法完成自动拆装箱,使用jad进行反编译可以看到该过程:

public class Demo {public static void main(String[] args) {int x = new Integer(10);  // 自动拆箱Integer y = x;            // 自动装箱}
}

反编译后结果:

public class Demo
{public Demo(){}public static void main(String args[]){int i = (new Integer(10)).intValue();   // intValue()拆箱Integer integer = Integer.valueOf(i);   // valueOf()装箱}
}

foreach语法糖

在遍历迭代时可以foreach语法糖,对于数组类型直接转换成for循环:

// 原始代码
int[] arr = {1, 2, 3, 4, 5};for(int item: arr) {System.out.println(item);}
}// 反编译后代码
int ai[] = {1, 2, 3, 4, 5
};
int ai1[] = ai;
int i = ai1.length;
// 转换成for循环
for(int j = 0; j < i; j++)
{int k = ai1[j];System.out.println(k);
}

对于容器类的遍历会使用iterator进行迭代:

import java.io.PrintStream;
import java.util.*;public class Demo
{public Demo() {}public static void main(String args[]){ArrayList arraylist = new ArrayList();arraylist.add(Integer.valueOf(1));arraylist.add(Integer.valueOf(2));arraylist.add(Integer.valueOf(3));Integer integer;// 使用的for循环+Iterator,类似于链表迭代:// for (ListNode cur = head; cur != null; System.out.println(cur.val)){//     cur = cur.next;// }for(Iterator iterator = arraylist.iterator(); iterator.hasNext(); System.out.println(integer))integer = (Integer)iterator.next();}
}

Arrays.asList(T…)

熟悉Arrays.asList(T…)用法的小伙伴都应该知道,asList()方法传入的参数不能是基本类型的数组,必须包装成包装类型再使用,否则对应生成的列表的大小永远是1:

import java.util.*;
public class Demo {public static void main(String[] args) {int[] arr1 = {1, 2, 3};Integer[] arr2 = {1, 2, 3};List lists1 = Arrays.asList(arr1);List lists2 = Arrays.asList(arr2);System.out.println(lists1.size()); // 1System.out.println(lists2.size()); // 3}
}

从反编译结果来解释,为什么传入基本类型的数组后,返回的List大小是1:

// 反编译后文件
import java.io.PrintStream;
import java.util.Arrays;
import java.util.List;public class Demo
{public Demo() {}public static void main(String args[]){int ai[] = {1, 2, 3};// 使用包装类型,全部元素由int包装为IntegerInteger ainteger[] = {Integer.valueOf(1), Integer.valueOf(2), Integer.valueOf(3)};// 注意这里被反编译成二维数组,而且是一个1行三列的二维数组// list.size()当然返回1List list = Arrays.asList(new int[][] { ai });List list1 = Arrays.asList(ainteger);System.out.println(list.size());System.out.println(list1.size());}
}

从上面结果可以看到,传入基本类型的数组后,会被转换成一个二维数组,而且是**new int[1][arr.length]**这样的数组,调用list.size()当然返回1。

注解

Java中的类、接口、枚举、注解都可以看做是类类型。使用jad来看一下@interface被转换成什么:

import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;@Retention(RetentionPolicy.RUNTIME)
public @interface Foo{String[] value();boolean bar();
}

查看反编译代码可以看出:

  • 自定义的注解类Foo被转换成接口Foo,并且继承Annotation接口
  • 原来自定义接口中的value()和bar()被转换成抽象方法
import java.lang.annotation.Annotation;public interface Fooextends Annotation
{public abstract String[] value();public abstract boolean bar();
}

注解通常和反射配合使用,而且既然自定义的注解最终被转换成接口,注解中的属性被转换成接口中的抽象方法,那么通过反射之后拿到接口实例,在通过接口实例自然能够调用对应的抽象方法:

import java.util.Arrays;@Foo(value={"sherman", "decompiler"}, bar=true)
public class Demo{public static void main(String[] args) {Foo foo = Demo.class.getAnnotation(Foo.class);System.out.println(Arrays.toString(foo.value())); // [sherman, decompiler]System.out.println(foo.bar());                    // true}
}

枚举

通过jad反编译可以很好地理解枚举类。

空枚举

先定义一个空的枚举类:

public enum DummyEnum {}

使用jad反编译查看结果:

  • 自定义枚举类被转换成final类,并且继承Enum
  • 提供了两个参数(name,odinal)的私有构造器,并且调用了父类的构造器。注意即使没有提供任何参数,也会有该该构造器,其中name就是枚举实例的名称,odinal是枚举实例的索引号
  • 初始化了一个private static final自定义类型的空数组 $VALUES
  • 提供了两个public static方法:
  • values()方法通过clone()方法返回内部$VALUES的浅拷贝。这个方法结合私有构造器可以完美实现单例模式,想一想values()方法是不是和单例模式中getInstance()方法功能类似
  • valueOf(String s):调用父类Enum的valueOf方法并强转返回
public final class DummyEnum extends Enum
{// 功能和单例模式的getInstance()方法相同public static DummyEnum[] values(){return (DummyEnum[])$VALUES.clone();}// 调用父类的valueOf方法,并墙砖返回public static DummyEnum valueOf(String s){return (DummyEnum)Enum.valueOf(DummyEnum, s);}// 默认提供一个私有的私有两个参数的构造器,并调用父类Enum的构造器private DummyEnum(String s, int i){super(s, i);}// 初始化一个private static final的本类空数组private static final DummyEnum $VALUES[] = new DummyEnum[0];}

包含抽象方法的枚举

枚举类中也可以包含抽象方法,但是必须定义枚举实例并且立即重写抽象方法,就像下面这样:

public enum DummyEnum {DUMMY1 {public void dummyMethod() {System.out.println("[1]: implements abstract method in enum class");}},DUMMY2 {public void dummyMethod() {System.out.println("[2]: implements abstract method in enum class");}};abstract void dummyMethod();}

再来反编译看看有哪些变化:

  • 原来final class变成了abstract class:这很好理解,有抽象方法的类自然是抽象类
  • 多了两个public static final的成员DUMMY1、DUMMY2,这两个实例的初始化过程被放到了static代码块中,并且实例过程中直接重写了抽象方法,类似于匿名内部类的形式。
  • 数组**$VALUES[]**初始化时放入枚举实例

还有其它变化么?

在反编译后的DummyEnum类中,是存在抽象方法的,而枚举实例在静态代码块中初始化过程中重写了抽象方法。在Java中,抽象方法和抽象方法重写同时放在一个类中,只能通过内部类形式完成。因此上面第二点应该说成就是以内部类形式初始化。

可以看一下DummyEnum.class存放的位置,应该多了两个文件:

  • DummyEnum$1.class
  • DummyEnum$2.class

Java中.class文件出现符号表示有内部类存在,就像OutClass符号表示有内部类存在,就像OutClass符号表示有内部类存在,就像OutClassInnerClass,这两个文件出现也应证了上面的匿名内部类初始化的说法。

import java.io.PrintStream;public abstract class DummyEnum extends Enum
{public static DummyEnum[] values(){return (DummyEnum[])$VALUES.clone();}public static DummyEnum valueOf(String s){return (DummyEnum)Enum.valueOf(DummyEnum, s);}private DummyEnum(String s, int i){super(s, i);}// 抽象方法abstract void dummyMethod();// 两个pubic static final实例public static final DummyEnum DUMMY1;public static final DummyEnum DUMMY2;private static final DummyEnum $VALUES[];// static代码块进行初始化static {DUMMY1 = new DummyEnum("DUMMY1", 0) {public void dummyMethod(){System.out.println("[1]: implements abstract method in enum class");}}
;DUMMY2 = new DummyEnum("DUMMY2", 1) {public void dummyMethod(){System.out.println("[2]: implements abstract method in enum class");}}
;// 对本类数组进行初始化$VALUES = (new DummyEnum[] {DUMMY1, DUMMY2});}
}

正常的枚举类

实际开发中,枚举类通常的形式是有两个参数(int code,Sring msg)的构造器,可以作为状态码进行返回。Enum类实际上也是提供了包含两个参数且是protected的构造器,这里为了避免歧义,将枚举类的构造器设置为三个,使用jad反编译:

最大的变化是:现在的private构造器从2个参数变成5个,而且在内部仍然将前两个参数通过super传递给父类,剩余的三个参数才是真正自己提供的参数。可以想象,如果自定义的枚举类只提供了一个参数,最终生成底层代码中private构造器应该有三个参数,前两个依然通过super传递给父类。

public final class CustomEnum extends Enum
{public static CustomEnum[] values(){return (CustomEnum[])$VALUES.clone();}public static CustomEnum valueOf(String s){return (CustomEnum)Enum.valueOf(CustomEnum, s);}private CustomEnum(String s, int i, int j, String s1, Object obj){super(s, i);code = j;msg = s1;data = obj;}public static final CustomEnum FIRST;public static final CustomEnum SECOND;public static final CustomEnum THIRD;private int code;private String msg;private Object data;private static final CustomEnum $VALUES[];static {FIRST = new CustomEnum("FIRST", 0, 10010, "first", Long.valueOf(100L));SECOND = new CustomEnum("SECOND", 1, 10020, "second", "Foo");THIRD = new CustomEnum("THIRD", 2, 10030, "third", new Object());$VALUES = (new CustomEnum[] {FIRST, SECOND, THIRD});}
}

JAD反编译tricks相关推荐

  1. JD-JUI 和Jad反编译 最新版1.6.6下载

    JD-JUI 最新1.6.6和Jad反编译工具 链接:https://pan.baidu.com/s/15qvNOEcebBFITeZP7HPulA 提取码:a9g3

  2. JAR包与JAD反编译

    Java开发过程中,不免会遇到使用第三方jar的情况.那么我们先来简单的说一下jar文件. JAR文件,全称为Java Archive File,就是Java档案文件.在开发过程中,许多专业的开发人员 ...

  3. eclipse中集成jad反编译

    [size=large]转载地址:[url]http://www.blogjava.net/sy1214520/archive/2009/04/29/209371.html[/url] 很早就听说过e ...

  4. 详解Eclipse集成Jad反编译工具

    一.什么是反编译? 高级语言源程序经过编译变成可执行文件,反编译就是逆过程,通俗的说,反编译是一个对目标可执行程序进行逆向分析,从而得到原始代码的过程,NET.Java这样的运行在虚拟机上的编程语言, ...

  5. jad环境变量配置_Java Jad 反编译

    下载运行文件 解压目录 添加环境变量,dos命令反编译class文件 这是JAD-快速Java反编译程序的自述文件. JAD主页:http://www.kpdus.com/jad.html 版权所有2 ...

  6. jad 反编译java

    1 安装 解压jad.zip文件到任何的目录.将会创建两个文件,一个是jad.exe另一个是readme文 件,不需要任何别安装 2 如何使用jad 如果我们有一个单独的java文件example1. ...

  7. myeclipse2013 jad反编译插件安装

    https://blog.csdn.net/zhenshixian88/article/details/26454319 1. 下载插件工具 http://download.csdn.net/deta ...

  8. myeclipse 安装jad反编译插件

    第一步: 下载jad.exe文件:http://www.varaneckas.com/jad 下载jadeclipse插件:http://sourceforge.net/projects/jadcli ...

  9. 在命令行下,如何使用JAD反编译jar文件

    在工作中,我们经常需要对遗留系统或其他的第三方系统进行维护,如果我们没有java源文件的情况下,一般都需要将其中的jar文件进行反编译. 本文将以如何用命令来反编译一个test.jar文件为例,你可以 ...

最新文章

  1. Luogu 2470 [SCOI2007]压缩
  2. Http持久连接与HttpClient连接池
  3. 安卓怎么用抖音做锁屏_把锁屏密码设置成当前时间,随时间的变动!别人怎么也猜不出来...
  4. 计算机利用公式计算实发工资怎么弄,2019新个税Excel计算器公式 助你轻松算出工资...
  5. Ubuntu通过windows代理上网
  6. AIX系统maxuproc太小导致ORA-27300错
  7. PDX模型不靠谱,土豪来发resource;细菌帮助癌转移,诺奖发现被推翻
  8. Java体系学习书籍推荐,BAT 面试官 如何面试
  9. 如何制作linux系统硬盘,手把手带你自制Linux系统之二 简易Linux的制作
  10. 今年双旦期间简直人品爆棚,晒晒我抽中的趣享付趣号卡
  11. 逐句回答,流式返回,ChatGPT采用的Server-sent events后端实时推送协议Python3.10实现,基于Tornado6.1
  12. Maven中央仓库地址
  13. 区块链投资入门篇—骨灰版
  14. 宇视NVR录像机 一直滴滴报警,如何关闭
  15. PHP.ini配置文件中文【转】
  16. 阿里云服务器ECS的用途有哪些?
  17. 百度副总裁李硕:通过“一企一档”等模式提升企业智能化水平
  18. 泛微e-cology9前台文件上传漏洞复现
  19. 看似“佛系”的手游《QQ炫舞》,背后的音频技术一点都不简单
  20. 删除cookie,domain的设置

热门文章

  1. (免费分享)基于springboot,vue在线考试系统
  2. 形态学在图像处理中的应用
  3. 独角兽项目 1 - 替罪羊
  4. Java入门 - 方法的使用
  5. 步进电机基础(7.6)-步进电机的选择方法-选择电机的计算方法
  6. 大数据分析了50万条拼多多商品数据, 得出了这样的结论
  7. Spring源码学习---------(最简单易懂的源码学习笔记)
  8. 线段树维护字符串哈希
  9. UTS 在安卓系统上的开发应用
  10. 从unpkg上下载资源