6

我用 Java 编写了一个实用程序方法:

public static final ImmutableSortedSet<TimeUnit> REVERSED_TIMEUNITS = ImmutableSortedSet.copyOf(
        Collections.<TimeUnit>reverseOrder(),
        EnumSet.allOf(TimeUnit.class)
);


/**
 * Get the number of ..., minutes, seconds and milliseconds
 *
 * You can specify a max unit so that you don't get days for exemple
 * and can get more than 24 hours if you want to display the result in hours
 *
 * The lowest unit is milliseconds
 * @param millies
 * @param maxTimeUnit
 * @return the result map with the higher unit first
 */
public static Map<TimeUnit,Long> getCascadingDateDiff(long millies,TimeUnit maxTimeUnit) {
    if ( maxTimeUnit == null ) {
        maxTimeUnit = TimeUnit.DAYS;
    }
    Map<TimeUnit,Long> map = new TreeMap<TimeUnit,Long>(Collections.<TimeUnit>reverseOrder());
    long restInMillies = millies;
    Iterable<TimeUnit> forUnits = REVERSED_TIMEUNITS.subSet(maxTimeUnit,TimeUnit.MICROSECONDS); // micros not included
    // compute the number of days, then number of hours, then minutes...
    for ( TimeUnit timeUnit : forUnits ) {
        long numberForUnit = timeUnit.convert(restInMillies,TimeUnit.MILLISECONDS);
        map.put(timeUnit,numberForUnit);
        restInMillies = restInMillies - timeUnit.toMillis(numberForUnit);
    }
    return map;
}

它适用于:

    Map<TimeUnit,Long> map = new TreeMap<TimeUnit,Long>(Collections.reverseOrder());

但我第一次尝试

    Map<TimeUnit,Long> map = Maps.newTreeMap(Collections.reverseOrder());

我的 IntelliJ 什么也没说,而我的编译器说:

DateUtils.java:[302,48] 不兼容的类型;不存在类型变量 K,V 的实例,因此 java.util.TreeMap 符合 java.util.Map [错误] 发现:java.util.TreeMap [错误] 需要:java.util.Map

没有比较器它工作正常:

   Map<TimeUnit,Long> map = Maps.newTreeMap();

但我尝试过:

Map<TimeUnit,Long> map = Maps.newTreeMap(Collections.<TimeUnit>reverseOrder());

与:

Map<TimeUnit,Long> map = Maps.newTreeMap(new Comparator<TimeUnit>() {
    @Override
    public int compare(TimeUnit timeUnit, TimeUnit timeUnit1) {
        return 0; 
    }
});

我得到了同样的错误。因此,似乎每次我在 TreeMap 中使用比较器时,类型推断都不再起作用。为什么?


Guava 方法的签名是:

  public static <C, K extends C, V> TreeMap<K, V> newTreeMap(Comparator<C> comparator)

预期的返回类型是没有比较器的类型,Java 能够推断出 K = TimeUnit 和 V = Long。

使用 TimeUnit 类型的比较器,Java 知道 C 是 TimeUnit。它还知道预期的返回类型是 K = TimeUnit 和 V = Long 的类型。K 扩展 C 受到尊重,因为 TimeUnit 扩展了 TimeUnit(无论如何,如果您认为它是错误的,我也尝试使用 Object 比较器......)

所以我只是想知道为什么类型推断在这种情况下不起作用?

4

1 回答 1

7

就像 Michael Laffargue 建议的那样,这是一个 OpenJDK6 类型推断错误:

https://bugs.openjdk.java.net/show_bug.cgi?id=100167

http://code.google.com/p/guava-libraries/issues/detail?id=635

它在我的 IntelliJ 中运行良好,在版本 7 中使用 OpenJDK,在版本 6 中使用其他 JDK。


kutschkem 的以下建议有效:

    Map<TimeUnit,Long> map = Maps.<TimeUnit,TimeUnit,Long>newTreeMap(Collections.<TimeUnit>reverseOrder());

注意<TimeUnit,TimeUnit,Long>which 允许显式地强制类型化参数。检查此相关主题:Java 中的泛型用法是什么?X.<Y>方法()

谢谢大家

于 2012-06-08T09:22:14.813 回答