Recently I've found a subtle difference between anonymous class and lambda expression:
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());
}
Usually lambdas are equivalent to the anonymous classes. Even my Eclipse IDE has the refactoring to convert the x
to lambda (it becomes exactly like y
) and convert y
to anonymous class (it becomes exactly like x
). However the lambda gives me a compilation error while anonymous class can be perfectly compiled. The error message looks like this:
>javac FinalTest.java
FinalTest.java:9: error: self-reference in initializer
final Runnable y = () -> System.out.println(y.hashCode());
^
1 error
So the question is: why there's such difference?
This has to do with JLS #8.3.3 dealing with forward references. In particular, if you use a fully qualified name it compiles (because the third condition of that rule becomes false 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());
In the case of the anonymous class, the fourth condition (C is the innermost class or interface enclosing the use) is not true because the enclosing class is the anonymous class itself.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With