从lambda表达式引用final字段

时间:2015-05-08 18:05:30

标签: java java-8 javac

最近我发现匿名类和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()); 
}

通常lambdas等同于匿名类。甚至我的Eclipse IDE都有重构将x转换为lambda(它变得与y完全相同)并将y转换为匿名类(它变得与x完全相同)。然而lambda给了我一个编译错误,而匿名类可以完美编译。错误消息如下所示:

>javac FinalTest.java
FinalTest.java:9: error: self-reference in initializer
    final Runnable y = () -> System.out.println(y.hashCode());
                                                ^
1 error

所以问题是:为什么会有这样的差异?

1 个答案:

答案 0 :(得分:15)

这与处理前向引用的JLS #8.3.3有关。特别是,如果您使用完全限定名称,则会编译(因为该规则的第三个条件变为false 在C的实例变量初始值设定项或C 的实例初始值设定项中,use是一个简单名称) :

final Runnable y = () -> System.out.println(this.y.hashCode());

对于匿名类,第四个条件( C是最内层的类或包含使用的接口)不正确,因为封闭类本身就是匿名类。