考虑以下类:
class Temp {
private final int field = 5;
int sum() {
return 1 + this.field;
}
}
然后我编译和反编译这个类:
> javac --version
javac 11.0.5
> javac Temp.java
> javap -v Temp.class
...
int sum();
descriptor: ()I
flags: (0x0000)
Code:
stack=2, locals=1, args_size=1
0: iconst_1
1: aload_0
2: invokestatic #3 // Method java/util/Objects.requireNonNull:(Ljava/lang/Object;)Ljava/lang/Object;
5: pop
6: iconst_5
7: iadd
8: ireturn
简单来说,javac
将 sum()
编译成:
int sum() {
final int n = 1;
Objects.requireNonNull(this); // <---
return n + 5;
}
这里的Objects.requireNonNull(this)
是做什么的?有什么意义吗?这与可达性有关吗?
Java 8编译器类似,它会插入this.getClass()
代替Objects.requireNonNull(this)
:
int sum() {
final int n = 1;
this.getClass(); // <---
return n + 5;
}
我也尝试使用Eclipse编译它。它没有插入requireNonNull
:
int sum() {
return 1 + 5;
}
所以这是javac特定的行为。
javac 15-ea
也可以观察到相同的行为。进一步观察,删除final
可以正常工作。 - Namanint sum();...
代码: 栈=2,本地变量=1,参数大小=1 0: iconst_1 1: aload_0 2: invokestatic #13 // Method java/util/Objects.requireNonNull:(Ljava/lang/Object;)Ljava/lang/Object; 5: pop 6: iconst_5 7: iadd 8: ireturn相对于删除最终关键字解析为
int sum();... 代码: 栈=2,本地变量=1,参数大小=1 0: iconst_1 1: aload_0 2: getfield #7 // Field field:I 5: iadd 6: ireturn` - Naman