在Java中重载方法时的奇怪行为

Sep*_*Sep 6 java

我今天遇到了这种奇怪的(在我看来)行为.拿这个简单的Test类:

public class Test {

public static void main(String[] args) {
    Test t = new Test();
    t.run();
}

private void run() {
    List<Object> list = new ArrayList<Object>();
    list.add(new Object());
    list.add(new Object());
    method(list);
}

public void method(Object o) {
    System.out.println("Object");
}

public void method(List<Object> o) {
    System.out.println("List of Objects");
}
}
Run Code Online (Sandbox Code Playgroud)

它的行为与您期望的一样,打印"对象列表".但是如果你改变以下三行:

List<String> list = new ArrayList<String>();
list.add("");
list.add("");
Run Code Online (Sandbox Code Playgroud)

你会得到"对象".

我尝试了其他一些方法并获得了相同的结果.这是一个错误还是正常行为?如果这是正常的,有人可以解释为什么吗?

谢谢.

axt*_*avt 12

这是正常的行为.List<String>不是List<Object>,所以method(Object)唯一适用的方法.

如果List<String>是a List<Object>,则可以违反类型安全性,例如,通过添加IntegerList<String>(并且由于类型擦除而无法在运行时捕获):

public void method(List<Object> o) { 
    o.add(new Integer(10));
}
Run Code Online (Sandbox Code Playgroud)

还要注意,数组具有不同的行为 - String[]是一个Object[],因为数组知道它的元素类型,如果你试图将错误的对象放入其中,则抛出运行时异常.


bra*_*ter 6

这是正常行为,因为当您使用泛型定义签名时,您可以指定单个类.(除非您使用通配符,否则链接会解释)...

所以List<String>不是List<Object>.然而,它是List<? extends Object>一个尝试,看看.