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()); // 1

System.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包装为Integer

Integer ainteger[] = {

Integer.valueOf(1), Integer.valueOf(2), Integer.valueOf(3)

};

// 注意这里被反编译成二维数组,而且是一个1行三列的二维数组

// list.size()当然返回1

List 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 Foo

extends 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$InnerClass,这两个文件出现也应证了上面的匿名内部类初始化的说法。

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

});

}

}

java反编译数组_JavaGuide/JAD反编译tricks.md at master · Github-Programer/JavaGuide · GitHub...相关推荐

  1. java p=x,Java-Tutorial/20、javac和javap.md at master · allenchenx/Java-Tutorial · GitHub

    目录 title: 夯实Java基础系列20:从IDE的实现原理聊起,谈谈那些年我们用过的Java命令 date: 2019-9-20 15:56:26 # 文章生成时间,一般不改 categorie ...

  2. jad 反编译java

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

  3. java反编译命令jad,java反编译小工具jad常用命令

    java反编译小工具jad常用命令 2014/6/11 19:39:24  sky52bc  程序员俱乐部  我要评论(0) 摘要:首先,看一下jad的帮助提示-a-用JVM字节格式来注解输出-af- ...

  4. java 反编译 class文件 jad eclipse乱码

    java 反编译 class文件 jad eclipse 编程 破解 工具 工具使用jad(还有joda,jd等工具) 1.使用jad反编译class文件,jad可以配置到eclipse中当做插件,直 ...

  5. JAD反编译tricks

    jad反编译工具,已经不再更新,且只支持JDK1.4,但并不影响其强大的功能. 基本用法:jad xxx.class,会生成直接可读的xxx.jad文件. 自动拆装箱 对于基本类型和包装类型之间的转换 ...

  6. java 反编译项目_Java 7 –反编译项目硬币

    java 反编译项目 大家好,该是从2012年开始写作的时候了.正如您在其他博客中可能已经看到的那样,有一些更改可以使您使用Java编程时的开发人员生活变得更加轻松:Diamond运算符,Switch ...

  7. JAR包与JAD反编译

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

  8. eclipse中集成jad反编译

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

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

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

  10. Mac 使用Jad 反编译class文件

    Mac下找到一个一款java反编译工具叫MacJAD,感觉是还是挺好用的.顺利就贴一下Jad的基本用法. 一.基本用法 Usage:jad [option(s)] 直接输入类文件名,且支持通配符,如下 ...

最新文章

  1. 【系列】EOS开发3 EOS提供的程序和工具
  2. 如何理解机器学习中的嵌入 (Embeddings)?
  3. import cycle not allowed_Cycle药物介绍醋酸群勃龙(2)
  4. 《敏捷开发绩效管理》扩展阅读(敏捷开发绩效管理,敏捷团队绩效管理)
  5. 一位小小码蚁工作2年多感想
  6. HDU - 6184 Counting Stars(思维+三元环)
  7. SAP Spartacus cxPageTemplateStyle Directive的作用
  8. 【转载】Linux关机命令详解
  9. python网站用什么数据库_PyMySQL数据库的使用
  10. 利用有名管道实现进程间的通信
  11. php_memcahed telnet远程操作方法
  12. Kali Linux 秘籍 第一章 安装和启动Kali
  13. HttpURLConnection请求数据流的写入(write)和读取(read)
  14. 空间直线同球体交点求解
  15. dede 表单必填_织梦给自定义表单增加必填功能,织梦表单必填设置
  16. Linux虚拟主机管理系统---wdcp
  17. E. Yet Another Array Counting Problem
  18. 分别用线性规划和动态规划求解打家劫舍问题(力扣198)
  19. 经典算法题:二分查找
  20. 新手如何运营好阿里国际站+产品标题组成

热门文章

  1. 惠普2t服务器硬盘,HP DL388 Gen8系列服务器硬盘超过2T的分区方法
  2. JRebel设置快捷键+激活方式
  3. 计算机主板 华硕 游戏用,DIY电脑的基石,华硕TUF B360M-PLUS GAMING S游戏主板
  4. 数据库基础介绍(1)几种常见的数据库
  5. 拼多多API SDK【拼多多开放平台】
  6. 西门子200PLC指令详解——比较指令
  7. 推荐两个高仿抖音 App 项目
  8. Wireshark实验 - DNS
  9. NB-IoT的优势是什么?
  10. DVP MIPI-CSI 摄像头接口的区别