代码之家  ›  专栏  ›  技术社区  ›  Robert Bain

这段代码使用ecj而不是javac进行编译。这是ecj、javac中的bug还是两者都不是?

  •  12
  • Robert Bain  · 技术社区  · 6 年前

    以下代码创建 Collector 产生一个 UnmodifiableSortedSet :

    package com.stackoverflow;
    
    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中的bug还是允许这两种行为的欠规范?

    Javac在java 9和10中的行为相同。

    2 回复  |  直到 6 年前
        1
  •  3
  •   Robert Bain    6 年前

    甲骨文已将此视为 compiler bug

        2
  •  2
  •   Jacob G.    6 年前

    有趣的是,它编译时不需要 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 对于这两种方法)。

    我进一步相信这就是原因,因为您可以定义泛型类型 T 为您的 class 并在两种方法中使用它(如果您删除 静止的 ):

    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);
        }
    }
    

    上面的编译和运行都很好。