此代码使用 ecj 而不是 javac 进行编译。这是 ecj、javac 中的错误还是两者都不是?
This code compiles using ecj but not javac. Is this a bug in ecj, javac or neither?
以下代码创建了一个 Collector
,它产生了一个 UnmodifiableSortedSet
:
package com.Whosebug;
import java.util.Collections;
import java.util.SortedSet;
import java.util.TreeSet;
import java.util.stream.Collector;
import java.util.stream.Collectors;
public class SOExample {
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toSortedSet() {
return Collectors.toCollection(TreeSet::new);
}
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(toSortedSet(), Collections::<T> unmodifiableSortedSet);
}
}
在ecj编译器下编译的代码:
$ java -jar ~/Downloads/ecj-3.13.101.jar -source 1.8 -target 1.8 SOExample.java
Picked up _JAVA_OPTIONS: -Duser.language=en -Duser.country=GB
然而在 javac 下:
$ javac -version
Picked up _JAVA_OPTIONS: -Duser.language=en -Duser.country=GB
javac 1.8.0_73
$ javac SOExample.java
Picked up _JAVA_OPTIONS: -Duser.language=en -Duser.country=GB
SOExample.java:16: error: method collectingAndThen in class Collectors cannot be applied to given types;
return Collectors.collectingAndThen(toSortedSet(), Collections::<T> unmodifiableSortedSet);
^
required: Collector<T#1,A,R>,Function<R,RR>
found: Collector<T#2,CAP#1,SortedSet<T#2>>,Collection[...]edSet
reason: cannot infer type-variable(s) T#3
(actual and formal argument lists differ in length)
where T#1,A,R,RR,T#2,T#3 are type-variables:
T#1 extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
A extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
R extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
RR extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
T#2 extends Comparable<T#2>
T#3 extends Object declared in method <T#3>unmodifiableSortedSet(SortedSet<T#3>)
where CAP#1 is a fresh type-variable:
CAP#1 extends Object from capture of ?
1 error
如果我将有问题的行更改为以下内容,代码将在两个编译器下编译:
return Collectors.collectingAndThen(toSortedSet(), (SortedSet<T> p) -> Collections.unmodifiableSortedSet(p));
这是 ecj、javac 中的错误还是允许这两种行为的规格不足?
Javac 在 java 9 和 10 中的行为相同。
有趣的是,它编译时不需要 toSortedSet
:
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(Collectors.toCollection(TreeSet::new), Collections::<T>unmodifiableSortedSet);
}
如果您显式将 T
传递给 toSortedSet
,它也会编译(删除 static
并使用 this.<T>toSortedSet()
也同样有效):
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(Test.<T>toSortedSet(), Collections::<T>unmodifiableSortedSet);
}
关于你为什么不按原样编译的问题,我怀疑这与两种方法之间的捕获类型不同有关,toSortedSet
需要完全相同的 T
在 toUnmodifiableSortedSet
中使用(因为您为这两种方法定义了通用类型 T
)。
我进一步相信这就是原因,因为您可以为您的 class
定义一个通用类型 T
并在两种方法中使用它(如果您删除 static
):
public class Test<T extends Comparable<? super T>> {
public static void main(String[] args) throws Exception {
System.out.println(Stream.of(5, 3, 4, 2, 1, 5, 4, 3, 2, 1)
.collect(new Test<Integer>().toUnmodifiableSortedSet()));
}
public Collector<T, ?, SortedSet<T>> toSortedSet() {
return Collectors.toCollection(TreeSet::new);
}
public Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(toSortedSet(), Collections::unmodifiableSortedSet);
}
}
上面的编译和运行都很好。
Oracle 已接受此作为 compiler bug。
以下代码创建了一个 Collector
,它产生了一个 UnmodifiableSortedSet
:
package com.Whosebug;
import java.util.Collections;
import java.util.SortedSet;
import java.util.TreeSet;
import java.util.stream.Collector;
import java.util.stream.Collectors;
public class SOExample {
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toSortedSet() {
return Collectors.toCollection(TreeSet::new);
}
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(toSortedSet(), Collections::<T> unmodifiableSortedSet);
}
}
在ecj编译器下编译的代码:
$ java -jar ~/Downloads/ecj-3.13.101.jar -source 1.8 -target 1.8 SOExample.java
Picked up _JAVA_OPTIONS: -Duser.language=en -Duser.country=GB
然而在 javac 下:
$ javac -version
Picked up _JAVA_OPTIONS: -Duser.language=en -Duser.country=GB
javac 1.8.0_73
$ javac SOExample.java
Picked up _JAVA_OPTIONS: -Duser.language=en -Duser.country=GB
SOExample.java:16: error: method collectingAndThen in class Collectors cannot be applied to given types;
return Collectors.collectingAndThen(toSortedSet(), Collections::<T> unmodifiableSortedSet);
^
required: Collector<T#1,A,R>,Function<R,RR>
found: Collector<T#2,CAP#1,SortedSet<T#2>>,Collection[...]edSet
reason: cannot infer type-variable(s) T#3
(actual and formal argument lists differ in length)
where T#1,A,R,RR,T#2,T#3 are type-variables:
T#1 extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
A extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
R extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
RR extends Object declared in method <T#1,A,R,RR>collectingAndThen(Collector<T#1,A,R>,Function<R,RR>)
T#2 extends Comparable<T#2>
T#3 extends Object declared in method <T#3>unmodifiableSortedSet(SortedSet<T#3>)
where CAP#1 is a fresh type-variable:
CAP#1 extends Object from capture of ?
1 error
如果我将有问题的行更改为以下内容,代码将在两个编译器下编译:
return Collectors.collectingAndThen(toSortedSet(), (SortedSet<T> p) -> Collections.unmodifiableSortedSet(p));
这是 ecj、javac 中的错误还是允许这两种行为的规格不足?
Javac 在 java 9 和 10 中的行为相同。
有趣的是,它编译时不需要 toSortedSet
:
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(Collectors.toCollection(TreeSet::new), Collections::<T>unmodifiableSortedSet);
}
如果您显式将 T
传递给 toSortedSet
,它也会编译(删除 static
并使用 this.<T>toSortedSet()
也同样有效):
public static <T extends Comparable<T>> Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(Test.<T>toSortedSet(), Collections::<T>unmodifiableSortedSet);
}
关于你为什么不按原样编译的问题,我怀疑这与两种方法之间的捕获类型不同有关,toSortedSet
需要完全相同的 T
在 toUnmodifiableSortedSet
中使用(因为您为这两种方法定义了通用类型 T
)。
我进一步相信这就是原因,因为您可以为您的 class
定义一个通用类型 T
并在两种方法中使用它(如果您删除 static
):
public class Test<T extends Comparable<? super T>> {
public static void main(String[] args) throws Exception {
System.out.println(Stream.of(5, 3, 4, 2, 1, 5, 4, 3, 2, 1)
.collect(new Test<Integer>().toUnmodifiableSortedSet()));
}
public Collector<T, ?, SortedSet<T>> toSortedSet() {
return Collectors.toCollection(TreeSet::new);
}
public Collector<T, ?, SortedSet<T>> toUnmodifiableSortedSet() {
return Collectors.collectingAndThen(toSortedSet(), Collections::unmodifiableSortedSet);
}
}
上面的编译和运行都很好。
Oracle 已接受此作为 compiler bug。