4

有没有办法使这种方法正确通用并消除警告?

/**
 * <p>Sort a collection by a certain "value" in its entries. This value is retrieved using
 * the given <code>valueFunction</code> which takes an entry as argument and returns
 * its value.</p>
 * 
 * <p>Example:</p>
 * <pre>// sort tiles by number
 *Collects.sortByValue(tileList, true, new Function<Integer,NormalTile>() {
 *  public Integer call(NormalTile t) {
 *      return t.getNumber();
 *  }
 *});</pre>
 *
 * @param list The collection.
 * @param ascending Whether to sort ascending (<code>true</code>) or descending (<code>false</code>).
 * @param valueFunction The function that retrieves the value of an entry.
 */
public static <T> void sortByValue(List<T> list, final boolean ascending, @SuppressWarnings("rawtypes") final Function<? extends Comparable, T> valueFunction) {
    Collections.sort(list, new Comparator<T>() {
        @SuppressWarnings({ "unchecked", "rawtypes" })
        @Override public int compare(T o1, T o2) {
            final Comparable v1 = valueFunction.call(o1);
            final Comparable v2 = valueFunction.call(o2);
            return v1.compareTo(v2) * (ascending ? 1 : -1);
        }
    });
}

我试过Function<? extends Comparable<?>, T>Function<? extends Comparable<? extends Comparable>, T>都没有编译,调用compareTo. 对于前者,即:

Comparable 类型中的方法 compareTo(capture#9-of ?) 不适用于参数(capture#10-of ? extends Comparable)

4

3 回答 3

5

试试这个:

public static <T, C extends Comparable<? super C>> void sortByValue(List<T> list, final boolean ascending, final Function<C, T> valueFunction) {
    Collections.sort(list, new Comparator<T>() {
        @Override public int compare(T o1, T o2) {
            final C v1 = valueFunction.apply(o1);
            final C v2 = valueFunction.apply(o2);
            return v1.compareTo(v2) * (ascending ? 1 : -1);
        }
    });
}

您还需要super允许为子类型定义比较器。更多解释:http: //docs.oracle.com/javase/tutorial/extra/generics/morefun.html

更新

此外,查看您的代码,我看到了另一辆自行车,Google Collections 有一个很好的库,它提供了非常方便的Ordering概念来处理它。

因此,您的代码如下所示:

Ordering<NormalTile> myOrdering = Ordering.natural()
  .onResultOf(new Function<Integer,NormalTile>() {
  public Integer call(NormalTile t) {
      return t.getNumber();
  }))
  .nullsLast();
...
Collections.sort(list, myOrdering);
//or
newList = myOrdering.sortedCopy(readonlyList);
于 2011-12-07T13:03:54.433 回答
2

这对我有用(Eclipse 编译器)

public static <T, U extends Comparable<U>> void sortByValue(
  List<T> list, final boolean ascending, final Function<U, T> valueFunction) {

  Collections.sort(list, new Comparator<T>() {
    @Override
    public int compare(T o1, T o2) {
      final U v1 = valueFunction.call(o1);
      final U v2 = valueFunction.call(o2);
      return v1.compareTo(v2) * (ascending ? 1 : -1);
    }
  });
}

正如其他人发布的那样,您甚至可以更进一步并声明U

U extends Comparable<? super U>

如果您有更多方法参数/返回值,这将派上用场,具体取决于U

于 2011-12-07T13:02:44.483 回答
1

如果为函数声明两个参数怎么办?

public static <T,C extends Comparable<C>> void sortByValue(List<T> list,
    final boolean ascending, final Function<C, T> valueFunction) {
...
final C v1 = ...
final C v2  ...

还没有用编译器检查自己的理智(没有你的接口,我太饿了无法模拟它们:)),但试一试。

我也太昏昏沉沉地无法推理它是否应该是C extends Comparable<C>C extends Comparable<? super C>。我认为前者会起作用并且更通用一些,尽管在实践中,除了针对自己之外,大多数类都没有实现 Comparable 。

于 2011-12-07T12:59:45.900 回答