20

我在我的大学里有高级编程课程,但我很难理解这段代码的工作原理。

public final class GenericClass<T> {
    private void overloadedMethod(Collection<?> o) {
        System.out.println("Collection<?>");
    }

    private void overloadedMethod(List<Number> o) {
        System.out.println("List<Number>");
    }

    private void overloadedMethod(ArrayList<Integer> o) {
        System.out.println("ArrayList<Integer>");
    }

    public void method(List<T> l) {
        overloadedMethod(l);
    }

    public static void main(String[] args) {
        GenericClass<Integer> test = new GenericClass<Integer>();
        test.method(new ArrayList<Integer>());
    }
}

为什么这段代码会打印“Collection<?>”?

4

3 回答 3

14

的声明method(List<T> l)没有指定类型 T 的任何界限。不能保证 T 是 Number 或 Number 的子类。因此,编译器只能决定该方法调用overloadedMethod(Collection<?> o).

请记住:编译后,类文件中不再提供有关泛型的信息。

于 2013-03-12T13:04:43.740 回答
-1

每当你定义一个泛型类型时,就会自动提供一个对应的原始类型这个方法

public void method(List<T> l) {

}

替换为

public void method(List<Object> l) {

}

如果您阅读了通配符类型,您将看到List<Number>and List<Object> or ArrayList<Integer>andList<Object>没有任何类型关系。List<Object>是 的子类型,Collecion<?>因此调用此方法。

于 2013-03-12T13:20:23.893 回答
-1

当您处理方法重载时,您必须牢记三个概念:

  1. 加宽
  2. 拳击
  3. 变量参数

您的问题在于扩大的第一点,java编译器为这些概念中的每一个提供与上面列出的相同顺序的优先级,因此它更喜欢扩大,当您调用您的方法时,test.method(new ArrayList<Integer>()); 编译器发现ArrayList<Integer>()可以扩大到Collection<?>和因为它具有最高优先级,所以它调用这些版本并忽略其他版本注意:如果您将其他两种方法的声明更改为private void overloadedMethod(ArrayList<?> o)private void overloadedMethod(List<?> o)编译器也会调用该Collection<?>版本,因为它是首选版本

于 2013-03-12T13:22:43.180 回答