You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

308 lines
12 KiB
Markdown

2 years ago
# 15 | Java语法糖与Java编译器
在前面的篇章中我们多次提到了Java语法和Java字节码的差异之处。这些差异之处都是通过Java编译器来协调的。今天我们便来列举一下Java编译器的协调工作。
## 自动装箱与自动拆箱
首先要提到的便是Java的自动装箱auto-boxing和自动拆箱auto-unboxing
我们知道Java语言拥有8个基本类型每个基本类型都有对应的包装wrapper类型。
之所以需要包装类型是因为许多Java核心类库的API都是面向对象的。举个例子Java核心类库中的容器类就只支持引用类型。
当需要一个能够存储数值的容器类时,我们往往定义一个存储包装类对象的容器。
对于基本类型的数值来说我们需要先将其转换为对应的包装类再存入容器之中。在Java程序中这个转换可以是显式也可以是隐式的后者正是Java中的自动装箱。
```
public int foo() {
ArrayList<Integer> list = new ArrayList<>();
list.add(0);
int result = list.get(0);
return result;
}
```
以上图中的Java代码为例。我构造了一个Integer类型的ArrayList并且向其中添加一个int值0。然后我会获取该ArrayList的第0个元素并作为int值返回给调用者。这段代码对应的Java字节码如下所示
```
public int foo();
Code:
0: new java/util/ArrayList
3: dup
4: invokespecial java/util/ArrayList."<init>":()V
7: astore_1
8: aload_1
9: iconst_0
10: invokestatic java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
13: invokevirtual java/util/ArrayList.add:(Ljava/lang/Object;)Z
16: pop
17: aload_1
18: iconst_0
19: invokevirtual java/util/ArrayList.get:(I)Ljava/lang/Object;
22: checkcast java/lang/Integer
25: invokevirtual java/lang/Integer.intValue:()I
28: istore_2
29: iload_2
30: ireturn
```
当向泛型参数为Integer的ArrayList添加int值时便需要用到自动装箱了。在上面字节码偏移量为10的指令中我们调用了Integer.valueOf方法将int类型的值转换为Integer类型再存储至容器类中。
```
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
```
这是Integer.valueOf的源代码。可以看到当请求的int值在某个范围内时我们会返回缓存了的Integer对象而当所请求的int值在范围之外时我们则会新建一个Integer对象。
在介绍反射的那一篇中我曾经提到参数java.lang.Integer.IntegerCache.high。这个参数将影响这里面的IntegerCache.high。
也就是说我们可以通过配置该参数扩大Integer缓存的范围。Java虚拟机参数-XX:+AggressiveOpts也会将IntegerCache.high调整至20000。
奇怪的是Java并不支持对IntegerCache.low的更改也就是说对于小于-128的整数我们无法直接使用由Java核心类库所缓存的Integer对象。
```
25: invokevirtual java/lang/Integer.intValue:()I
```
当从泛型参数为Integer的ArrayList取出元素时我们得到的实际上也是Integer对象。如果应用程序期待的是一个int值那么就会发生自动拆箱。
在我们的例子中自动拆箱对应的是字节码偏移量为25的指令。该指令将调用Integer.intValue方法。这是一个实例方法直接返回Integer对象所存储的int值。
## 泛型与类型擦除
你可能已经留意到了在前面例子生成的字节码中往ArrayList中添加元素的add方法所接受的参数类型是Object而从ArrayList中获取元素的get方法其返回类型同样也是Object。
前者还好但是对于后者在字节码中我们需要进行向下转换将所返回的Object强制转换为Integer方能进行接下来的自动拆箱。
```
13: invokevirtual java/util/ArrayList.add:(Ljava/lang/Object;)Z
...
19: invokevirtual java/util/ArrayList.get:(I)Ljava/lang/Object;
22: checkcast java/lang/Integer
```
之所以会出现这种情况是因为Java泛型的类型擦除。这是个什么概念呢简单地说那便是Java程序里的泛型信息在Java虚拟机里全部都丢失了。这么做主要是为了兼容引入泛型之前的代码。
当然并不是每一个泛型参数被擦除类型后都会变成Object类。对于限定了继承类的泛型参数经过类型擦除后所有的泛型参数都将变成所限定的继承类。也就是说Java编译器将选取该泛型所能指代的所有类中层次最高的那个作为替换泛型的类。
```
class GenericTest<T extends Number> {
T foo(T t) {
return t;
}
}
```
举个例子在上面这段Java代码中我定义了一个T extends Number的泛型参数。它所对应的字节码如下所示。可以看到foo方法的方法描述符所接收参数的类型以及返回类型都为Number。方法描述符是Java虚拟机识别方法调用的目标方法的关键。
```
T foo(T);
descriptor: (Ljava/lang/Number;)Ljava/lang/Number;
flags: (0x0000)
Code:
stack=1, locals=2, args_size=2
0: aload_1
1: areturn
Signature: (TT;)TT;
```
不过字节码中仍存在泛型参数的信息如方法声明里的T foo(T)以及方法签名Signature中的“(TT;)TT;”。这类信息主要由Java编译器在编译他类时使用。
既然泛型会被类型擦除,那么我们还有必要用它吗?
我认为是有必要的。Java编译器可以根据泛型参数判断程序中的语法是否正确。举例来说尽管经过类型擦除后ArrayList.add方法所接收的参数是Object类型但是往泛型参数为Integer类型的ArrayList中添加字符串对象Java编译器是会报错的。
```
ArrayList<Integer> list = new ArrayList<>();
list.add("0"); // 编译出错
```
## 桥接方法
泛型的类型擦除带来了不少问题。其中一个便是方法重写。在第四篇的课后实践中,我留了这么一段代码:
```
class Merchant<T extends Customer> {
public double actionPrice(T customer) {
return 0.0d;
}
}
class VIPOnlyMerchant extends Merchant<VIP> {
@Override
public double actionPrice(VIP customer) {
return 0.0d;
}
}
```
VIPOnlyMerchant中的actionPrice方法是符合Java语言的方法重写的毕竟都使用@Override来注解了。然而经过类型擦除后父类的方法描述符为(LCustomer;)D而子类的方法描述符为(LVIP;)D。这显然不符合Java虚拟机关于方法重写的定义。
为了保证编译而成的Java字节码能够保留重写的语义Java编译器额外添加了一个桥接方法。该桥接方法在字节码层面重写了父类的方法并将调用子类的方法。
```
class VIPOnlyMerchant extends Merchant<VIP>
...
public double actionPrice(VIP);
descriptor: (LVIP;)D
flags: (0x0001) ACC_PUBLIC
Code:
0: dconst_0
1: dreturn
public double actionPrice(Customer);
descriptor: (LCustomer;)D
flags: (0x1041) ACC_PUBLIC, ACC_BRIDGE, ACC_SYNTHETIC
Code:
0: aload_0
1: aload_1
2: checkcast class VIP
5: invokevirtual actionPrice:(LVIP;)D
8: dreturn
// 这个桥接方法等同于
public double actionPrice(Customer customer) {
return actionPrice((VIP) customer);
}
```
在我们的例子中VIPOnlyMerchant类将包含一个桥接方法actionPrice(Customer)它重写了父类的同名同方法描述符的方法。该桥接方法将传入的Customer参数强制转换为VIP类型再调用原本的actionPrice(VIP)方法。
当一个声明类型为Merchant实际类型为VIPOnlyMerchant的对象调用actionPrice方法时字节码里的符号引用指向的是Merchant.actionPrice(Customer)方法。Java虚拟机将动态绑定至VIPOnlyMerchant类的桥接方法之中并且调用其actionPrice(VIP)方法。
需要注意的是在javap的输出中该桥接方法的访问标识符除了代表桥接方法的ACC\_BRIDGE之外还有ACC\_SYNTHETIC。它表示该方法对于Java源代码来说是不可见的。当你尝试通过传入一个声明类型为Customer的对象作为参数调用VIPOnlyMerchant类的actionPrice方法时Java编译器会报错并且提示参数类型不匹配。
```
Customer customer = new VIP();
new VIPOnlyMerchant().actionPrice(customer); // 编译出错
```
当然,如果你实在想要调用这个桥接方法,那么你可以选择使用反射机制。
```
class Merchant {
public Number actionPrice(Customer customer) {
return 0;
}
}
class NaiveMerchant extends Merchant {
@Override
public Double actionPrice(Customer customer) {
return 0.0D;
}
}
```
除了前面介绍的泛型重写会生成桥接方法之外如果子类定义了一个与父类参数类型相同的方法其返回类型为父类方法返回类型的子类那么Java编译器也会为其生成桥接方法。
```
class NaiveMerchant extends Merchant
public java.lang.Double actionPrice(Customer);
descriptor: (LCustomer;)Ljava/lang/Double;
flags: (0x0001) ACC_PUBLIC
Code:
stack=2, locals=2, args_size=2
0: dconst_0
1: invokestatic Double.valueOf:(D)Ljava/lang/Double;
4: areturn
public java.lang.Number actionPrice(Customer);
descriptor: (LCustomer;)Ljava/lang/Number;
flags: (0x1041) ACC_PUBLIC, ACC_BRIDGE, ACC_SYNTHETIC
Code:
stack=2, locals=2, args_size=2
0: aload_0
1: aload_1
2: invokevirtual actionPrice:(LCustomer;)Ljava/lang/Double;
5: areturn
```
我之前曾提到过class文件里允许出现两个同名、同参数类型但是不同返回类型的方法。这里的原方法和桥接方法便是其中一个例子。由于该桥接方法同样标注了ACC\_SYNTHETIC因此当在Java程序中调用NaiveMerchant.actionPrice时我们只会调用到原方法。
## 其他语法糖
在前面的篇章中我已经介绍过了变长参数、try-with-resources以及在同一catch代码块中捕获多种异常等语法糖。下面我将列举另外两个常见的语法糖。
foreach循环允许Java程序在for循环里遍历数组或者Iterable对象。对于数组来说foreach循环将从0开始逐一访问数组中的元素直至数组的末尾。其等价的代码如下面所示
```
public void foo(int[] array) {
for (int item : array) {
}
}
// 等同于
public void bar(int[] array) {
int[] myArray = array;
int length = myArray.length;
for (int i = 0; i < length; i++) {
int item = myArray[i];
}
}
```
对于Iterable对象来说foreach循环将调用其iterator方法并且用它的hasNext以及next方法来遍历该Iterable对象中的元素。其等价的代码如下面所示
```
public void foo(ArrayList<Integer> list) {
for (Integer item : list) {
}
}
// 等同于
public void bar(ArrayList<Integer> list) {
Iterator<Integer> iterator = list.iterator();
while (iterator.hasNext()) {
Integer item = iterator.next();
}
}
```
字符串switch编译而成的字节码看起来非常复杂但实际上就是一个哈希桶。由于每个case所截获的字符串都是常量值因此Java编译器会将原来的字符串switch转换为int值switch比较所输入的字符串的哈希值。
由于字符串哈希值很容易发生碰撞因此我们还需要用String.equals逐个比较相同哈希值的字符串。
如果你感兴趣的话可以自己利用javap分析字符串switch编译而成的字节码。
## 总结与实践
今天我主要介绍了Java编译器对几个语法糖的处理。
基本类型和其包装类型之间的自动转换,也就是自动装箱、自动拆箱,是通过加入\[Wrapper\].valueOf如Integer.valueOf以及\[Wrapper\].\[primitive\]Value如Integer.intValue方法调用来实现的。
Java程序中的泛型信息会被擦除。具体来说Java编译器将选取该泛型所能指代的所有类中层次最高的那个作为替换泛型的具体类。
由于Java语义与Java字节码中关于重写的定义并不一致因此Java编译器会生成桥接方法作为适配器。此外我还介绍了foreach循环以及字符串switch的编译。
今天的实践环节你可以探索一下Java 10的var关键字是否保存了泛型信息是否支持自动装拆箱
```
public void foo() {
var value = 1;
var list = new ArrayList<Integer>();
list.add(value);
// list.add("1"); 这一句能够编译吗?
}
```