1
0
mirror of https://github.com/Snailclimb/JavaGuide synced 2025-06-16 18:10:13 +08:00

372 lines
11 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

# JAD 反编译
[jad](https://varaneckas.com/jad/)反编译工具,已经不再更新,且只支持 JDK1.4,但并不影响其强大的功能。
基本用法:`jad xxx.class`,会生成直接可读的 `xxx.jad` 文件。
## 自动拆装箱
对于基本类型和包装类型之间的转换,通过 xxxValue()和 valueOf()两个方法完成自动拆装箱,使用 jad 进行反编译可以看到该过程:
```java
public class Demo {
public static void main(String[] args) {
int x = new Integer(10); // 自动拆箱
Integer y = x; // 自动装箱
}
}
```
反编译后结果:
```java
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 循环:
```java
// 原始代码
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 进行迭代:
```java
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
```java
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
```java
// 反编译后文件
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 被转换成什么:
```java
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()被转换成抽象方法
```java
import java.lang.annotation.Annotation;
public interface Foo
extends Annotation
{
public abstract String[] value();
public abstract boolean bar();
}
```
注解通常和反射配合使用,而且既然自定义的注解最终被转换成接口,注解中的属性被转换成接口中的抽象方法,那么通过反射之后拿到接口实例,在通过接口实例自然能够调用对应的抽象方法:
```java
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 反编译可以很好地理解枚举类。
### 空枚举
先定义一个空的枚举类:
```java
public enum DummyEnum {
}
```
使用 jad 反编译查看结果:
- 自定义枚举类被转换成 final 类,并且继承 Enum
- 提供了两个参数nameodinal的私有构造器并且调用了父类的构造器。注意即使没有提供任何参数也会有该构造器其中 name 就是枚举实例的名称odinal 是枚举实例的索引号
- 初始化了一个 private static final 自定义类型的空数组 **\$VALUES**
- 提供了两个 public static 方法:
- values()方法通过 clone()方法返回内部\$VALUES 的浅拷贝。这个方法结合私有构造器可以完美实现单例模式,想一想 values()方法是不是和单例模式中 getInstance()方法功能类似
- valueOf(String s):调用父类 Enum 的 valueOf 方法并强转返回
```java
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];
}
```
### 包含抽象方法的枚举
枚举类中也可以包含抽象方法,但是必须定义枚举实例并且立即重写抽象方法,就像下面这样:
```java
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这两个文件出现也应证了上面的匿名内部类初始化的说法。
```java
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 codeSring msg的构造器可以作为状态码进行返回。Enum 类实际上也是提供了包含两个参数且是 protected 的构造器,这里为了避免歧义,将枚举类的构造器设置为三个,使用 jad 反编译:
最大的变化是:现在的 private 构造器从 2 个参数变成 5 个,而且在内部仍然将前两个参数通过 super 传递给父类,剩余的三个参数才是真正自己提供的参数。可以想象,如果自定义的枚举类只提供了一个参数,最终生成底层代码中 private 构造器应该有三个参数,前两个依然通过 super 传递给父类。
```java
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
});
}
}
```