17

假设你有一个方法

  • 接受一个阈值和一个输入
  • 如果输入小于阈值,则引发异常
  • 否则返回输入

它看起来像这样:

<N extends Number & Comparable<N>, S extends N> S ensureLessThan(N threshold, S input) {
    if (input.compareTo(threshold) >= 0) {
        throw new IllegalArgumentException("Input " + input + " is not less than " + threshold);
    }
    return input;
}

运行时,此方法会抛出NoSuchMethodError

Exception in thread "main" java.lang.NoSuchMethodError: java.lang.Number.compareTo(Ljava/lang/Object;)I

添加看起来像多余的演员表使其工作:

...
    if (((N) input).compareTo(threshold) >= 0) {
...

那么这里发生了什么?

更新:我的 Java 版本是

java version "1.6.0_37"
Java(TM) SE Runtime Environment (build 1.6.0_37-b06-434-11M3909)
Java HotSpot(TM) 64-Bit Server VM (build 20.12-b01-434, mixed mode)

这是一个可运行的示例:https ://gist.github.com/4526536

4

3 回答 3

7

我怀疑这是编译器中的错误。它已经静态绑定到Number.compareTo(Object)不存在的,当它应该是Comparable.compareTo(Object)

于 2013-01-13T21:23:23.447 回答
5
if (threshold.compareTo(input) < 0) {

将工作。

这显然是编译器错误。这样做的原因是编译器生成了一个可比较的检查强制转换。而反之则不然。

ensureLessThan(Ljava/lang/Number;Ljava/lang/Number;)Ljava/lang/Number;
  L0
        LINENUMBER 11 L0
        ALOAD 1
        CHECKCAST java/lang/Comparable
        ALOAD 2
        INVOKEINTERFACE java/lang/Comparable.compareTo (Ljava/lang/Object;)I
        IFGE L1

相对

ensureLessThan(Ljava/lang/Number;Ljava/lang/Number;)Ljava/lang/Number;
        LINENUMBER 11 L0
        ALOAD 1
        ALOAD 2
        INVOKEVIRTUAL java/lang/Number.compareTo (Ljava/lang/Object;)I
        IFGE L1
于 2013-01-13T22:00:34.357 回答
2

这很奇怪,以下代码从我这边编译并运行:

/**
 * @author Buhake Sindi
 * @since 14 January 2013
 *
 */
public class Test {

    public static <N extends Number & Comparable<N>, S extends N> S ensureLessThan(N threshold, S input) {
        if (input.compareTo(threshold) >= 0) {
            throw new IllegalArgumentException("Input " + input + " is not less than " + threshold);
        }
        return input;
    }

    public static void main(String[] args) {
        ensureLessThan(10, 5);
    }
}

在 Eclipse Juno (Java EE) 中测试,使用以下 JDK:

  1. JDK 1.6.0_21。
  2. JDK 1.7.0_09-b05
于 2013-01-13T21:30:56.943 回答