I ran into a compilation failure while writing some Java code, which I distilled down to the following test case:
import java.util.Collections;
import java.util.List;
public class TernaryFailure {
public static List<String> thisWorks() {
return Collections.emptyList();
}
public static List<String> thisFailsToCompile() {
return true ? Collections.emptyList() : Collections.emptyList();
}
}
The code above fails to compile with javac
with JDK 1.7.0_45:
$ javac TernaryFailure.java TernaryFailure.java:10: error: incompatible types return true ? Collections.emptyList() : Collections.emptyList(); ^ required: List<String> found: List<Object> 1 error
However, it compiles without any error with JDK 1.8.0_05.
Is that a bug in the Java 7 implementation? Or was there an enhancement to the Java Language Specification in Java 8 to start allowing this — and if so, what was the change?
The JLS SE 8 says at (§15.2):
So from this part of the spec is clear that conditional expressions, the ternary operator, can be considered poly expressions. But not all conditional expressions can be considered poly expressions, only reference conditional expressions according to (§15.25). The conditions under which a reference conditional expression can be considered a poly expression are clarified at (§15.25.3):
Check that in your example the conditional expression appears in an assignment context because according to (§14.17):
So at the end of the day, what all this means? This implied that when conditional expressions are poly expressions, the target type is "pushed down" to each operand. This way the compiler can attribute each part of the condition, against the target. In your case the target is
List<String>
. If we check the definition of the emptyList() method we have:So with the target
List<String>
, the compiler can infer that T == String and the code is accepted.