During the migration of our code base from java 1.7 to 1.8 we’ve got an error message "The method ... is not applicable for the arguments" on several code locations, all following the same pattern in generics usage.
We are currently using mostly Eclipse Mars
(4.5.2) on Windows 7
, but could confirm the behavior with Neon
(4.6), too. Javac
as well as ecj
with 1.7 compliance level can both compile our code without an error.
Here is a Minimal, Complete, and Verifiable example:
public class ComplexInterfaceTest {
public static class Foo {}
public interface Bar {
void print();
}
public static class SubFooBar extends Foo implements Bar {
public void print() {
System.out.println(this.getClass().getSimpleName());
}
}
public static class FooBar<T extends Foo & Bar> {
public static <T extends Foo & Bar> FooBar<T> makeFooBar() {
return new FooBar<>();
}
public void create(T value) {
value.print();
return;
}
}
public static class Base<T extends Foo> {}
public static class Subclass extends Base<SubFooBar> {
public void doSomething(SubFooBar value) {
// FooBar.<SubFooBar>makeFooBar().create(value);
FooBar.makeFooBar().create(value);
}
}
public static void main(String[] args) {
new Subclass().doSomething(new SubFooBar());
}
}
Now switching the commented out lines in doSomething
method makes the code to compile, so we do have a workaround. Still the error message seems not right, as the class SubFooBar
extend Foo
and implements Bar
, so it fulfills the contract of <T extends Foo & Bar>
, which is required in <T extends Foo & Bar> FooBar<T> makeFooBar()
, so actually T
IMO should be bound to SubFooBar
.
I searched for similar questions and found these: Differences in type inference JDK8 javac/Eclipse Luna? Type Inference Compiler Error In Eclipse with Java8 but not with Java7
Which makes me think it might be an ecj
bug. In this course I also looked into Eclipse Bugzilla
but couldn’t find anything comparable, I saw these:
Now Eclipse Bugzilla
discussions are full of details about internal workings of the ecj
, which I not always can follow. What I understand though is the general consensus there, that Eclipse
compiler has to strictly follow JLS
and not javac
(in cases where it were wrong), so it doesn’t necessarily must be a bug in ecj
. If it weren’t an ecj
bug, then compiling the code must have been a javac
bug.
What I am interested in is – for those that can analyze the type inference process of my code snippet - should the code have compiled or did I make error in coding?
EDIT
As I promised to post the outcome of my report to Eclipse’s Bugzilla
: the defect has the ID #497905 (Stephan Herrmann has posted the link in his comment below the accepted answer) and is currently targeted for v4.7.
In the method
public void doSomething(SubFooBar value) {
FooBar.makeFooBar().create(value);
}
the type parameter T
of the method makeFooBar()
will never be inferred as SubFooBar
. The fact that you are going to pass an instance of SubFooBar
to the create
method afterwards, does not influence the type of the preceding invocation expression FooBar.makeFooBar()
.
This doesn’t change with Java 8’s target typing, as this new feature doesn’t work on the receiver expression of a chained method invocation.
So in all versions, the type inferred for T
of the makeFooBar()
invocation will be the intersection type Foo & Bar
, so the result type is FooBar<Foo&Bar>
. This is also what Eclipse infers, even if the tooltip in the editor may display something else.
This implies that you can pass a SubFooBar
instance to the create
method as FooBar<Foo&Bar>.create(…)
expects an instance of Foo&Bar
and SubFooBar
extending Foo
and implementing Bar
is compatible.
It’s possible to demonstrate that Eclipse does infer the same type as all other compilers, as inserting an appropriate type cast
public void doSomething(SubFooBar value) {
FooBar.makeFooBar().create((Foo&Bar)value);
}
makes the compiler error go away. So the problem here is not the type inference, but that this Eclipse version thinks that SubFooBar
is not assignable to Foo & Bar
.