引用 lambda 表达式的最终字段
Reference to the final field from lambda expression
最近我发现了匿名 class 和 lambda 表达式之间的细微差别:
public class FinalTest {
final Runnable x = new Runnable() {
@Override
public void run() {
System.out.println(x.hashCode());
}
};
final Runnable y = () -> System.out.println(y.hashCode());
}
通常 lambda 等同于匿名 classes。甚至我的 Eclipse IDE 也进行了重构以将 x
转换为 lambda(它变得完全像 y
)并将 y
转换为匿名 class(它变得完全一样像 x
)。但是 lambda 给了我一个编译错误,而匿名 class 可以完美编译。错误消息如下所示:
>javac FinalTest.java
FinalTest.java:9: error: self-reference in initializer
final Runnable y = () -> System.out.println(y.hashCode());
^
1 error
所以问题是:为什么会有这样的差异?
这与 JLS #8.3.3 处理前向引用有关。特别是,如果您使用完全限定名称,它会编译(因为该规则的第三个条件变为假 The use is a simple name in either an instance variable initializer of C or an instance initializer of C):
final Runnable y = () -> System.out.println(this.y.hashCode());
在匿名class的情况下,第四个条件(C是最里面的class或包含使用的接口)不成立,因为封闭的 class 是匿名 class 本身。
最近我发现了匿名 class 和 lambda 表达式之间的细微差别:
public class FinalTest {
final Runnable x = new Runnable() {
@Override
public void run() {
System.out.println(x.hashCode());
}
};
final Runnable y = () -> System.out.println(y.hashCode());
}
通常 lambda 等同于匿名 classes。甚至我的 Eclipse IDE 也进行了重构以将 x
转换为 lambda(它变得完全像 y
)并将 y
转换为匿名 class(它变得完全一样像 x
)。但是 lambda 给了我一个编译错误,而匿名 class 可以完美编译。错误消息如下所示:
>javac FinalTest.java
FinalTest.java:9: error: self-reference in initializer
final Runnable y = () -> System.out.println(y.hashCode());
^
1 error
所以问题是:为什么会有这样的差异?
这与 JLS #8.3.3 处理前向引用有关。特别是,如果您使用完全限定名称,它会编译(因为该规则的第三个条件变为假 The use is a simple name in either an instance variable initializer of C or an instance initializer of C):
final Runnable y = () -> System.out.println(this.y.hashCode());
在匿名class的情况下,第四个条件(C是最里面的class或包含使用的接口)不成立,因为封闭的 class 是匿名 class 本身。