The following class contains a member variable runnable
which is initialized with an instance of an anonymous inner class. The inner class references the same member:
class Example {
Runnable runnable = new Runnable() {
@Override
public void run() {
System.out.println(runnable);
}
};
}
This is not a problem as long as the method is not executed before the member has been assigned and the JLS allows such a reference.
The declaration of the member variable could theoretically be converted to a lambda expression like this:
Runnable runnable = () -> System.out.println(runnable);
To my understanding, this is functionally equivalent to the previous example, but it is rejected by javac 1.8.0_05
with the following error message:
Error:(2, 54) java: self-reference in initializer
While that statement is true, I do not see why this was disallowed. Was this intentionally disallowed, maybe because lambda expressions are compiled to different byte code which would lead to problems if it was allows? Or was just disallowed because there were already problems with these references when they were used in anonymous inner classes? Or was it unintentionally disallowed by the JLS writers? Or is it a bug in javac
?
Bug #JDK-8027941 describes exactly this. Dan Smith (Project Lambda Specification Lead) writes that it's not a bug, and not limited to lambdas.
In the comments on a related bug report, he puts it like this:
He also remarks:
I think the problem is that the designers of Java want to have simple, syntactic rules to decide what kind of statements are allowed, rather than depending on more complex, semantic code analysis. The benefit is probably a simpler spec and therefore less requirements on compilers, while the cost is that programmers can't express every program -- at least not in the way they want to.
As Marko Topolnik points out, there is a solution: fully qualify the field. Example from the bug report: