我在大学开设了高级编程课程,我对理解这段代码的工作方式有点麻烦。
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&lt;?&gt;”?
答案 0 :(得分:14)
method(List<T> l)
的声明未指定类型T的任何边界。不能保证T是Number或Number的子类。因此,编译器只能决定此方法调用overloadedMethod(Collection<?> o)
。
请记住:编译后,类文件中不再提供有关泛型的信息。
答案 1 :(得分:-1)
每当定义泛型类型时,都会自动提供相应的原始类型 这个方法
public void method(List<T> l) {
}
正在替换为
public void method(List<Object> l) {
}
如果您阅读有关通配符类型的信息,您会看到List<Number>
和List<Object>
或ArrayList<Integer>
和List<Object>
没有任何类型关系。 List<Object>
是Collecion<?>
的子类型,因此调用此方法。
答案 2 :(得分:-1)
当你处理方法重载时,你必须在脑海中放置三个概念:
你的问题在于第一点正在扩大,java编译器为每个概念提供了与上面列出的顺序相同的优先级,因此它更喜欢扩展,当你将方法称为test.method(new ArrayList<Integer>());
时编译器发现ArrayList<Integer>()
可以扩展为Collection<?>
,因为它具有最高优先级,所以它调用这些版本并忽略其他版本
注意:如果您将其他两种方法的声明更改为private void overloadedMethod(ArrayList<?> o)
和private void overloadedMethod(List<?> o)
,编译器也会调用Collection<?>
版本,因为它是首选版本