List和List<之间有什么区别?>, List<T>, List<E>, List<Object>?

1. 列表

List:是一个原始类型,因此不是类型安全的。它只会在强制转换错误时生成运行时错误。当强制转换错误时,我们希望出现编译时错误。不建议使用。

2. 列表< ?>

列表< ?>是一个无界通配符。但我不确定这是干什么用的?我可以打印一个List<?>无问题:

public static void test(List<?> list){
    System.out.println(list);   // Works
}

为什么我不能向List<?>?

public static void test(List<?> list){
    list.add(new Long(2));     // Error
    list.add("2");             // Error
    System.out.println(list);
}

3. 目前居住;rolelabel T >

public static void test(List<T> list){   // T cannot be resolved
    System.out.println(list);
}

我不明白这个语法。我看到了类似这样的东西,它起作用了:

public <T> T[] toArray(T[] a){
    return a;   
}

有时,我看到<T>,或<E>,或<U>, <T,E>。它们都是一样的还是代表着不同的东西?

4. List <物体>

这给出了错误“the method test(List<Object>) is not applied for the argument List<String>”:

public static void test(List<Object> list){
    System.out.println(list);
}

如果我尝试这样做,那么我得到“不能从列表<字符串>到列表<对象>”:

test((List<Object>) names);

我很困惑。字符串是对象的子类,所以为什么List<String>不是List<Object>的子类?


当前回答

问题2没问题,因为“System.out.println(set);”意味着“System.out.println(set. tostring());”set是List的一个实例,因此编译器将调用List. tostring ();

public static void test(List<?> set){
set.add(new Long(2)); //--> Error  
set.add("2");    //--> Error
System.out.println(set);
} 
Element ? will not promise Long and String, so complier will  not accept Long and String Object

public static void test(List<String> set){
set.add(new Long(2)); //--> Error
set.add("2");    //--> Work
System.out.println(set);
}
Element String promise it a String, so complier will accept String Object

问题3:这些符号是一样的,但是你可以给它们不同的规格。例如:

public <T extends Integer,E extends String> void p(T t, E e) {}

问题4:集合不允许类型参数协方差。但是数组允许协方差。

其他回答

你是对的:String是Object的子集。由于String比Object更“精确”,您应该强制转换它以将其用作System.out.println()的参数。

在你的第三点,“T”不能解析,因为它没有声明,通常当你声明一个泛型类时,你可以使用“T”作为绑定类型参数的名称,许多在线示例包括oracle的教程使用“T”作为类型参数的名称,例如,你声明一个类:

public class FooHandler<T>
{
   public void operateOnFoo(T foo) { /*some foo handling code here*/}

}

你是说,FooHandler的operateOnFoo方法期望一个类型为“T”的变量在类声明本身声明,考虑到这一点,你可以稍后添加另一个方法,如

public void operateOnFoos(List<T> foos)

在所有情况下,T, E或U都是类型参数的标识符,你甚至可以有多个使用该语法的类型参数

public class MyClass<Atype,AnotherType> {}

在你的第四点,虽然实际上Sting是对象的子类型,在泛型类中没有这样的关系,List<String>不是List<Object>的子类型,从编译器的角度来看,它们是两种不同的类型,这是最好的解释在这个博客条目

我建议阅读Java谜题。它很好地解释了声明中的继承、泛型、抽象和通配符。 http://www.javapuzzlers.com/

符号List<?>的意思是“一个列表的东西(但我不是说什么)”。因为测试中的代码适用于列表中的任何类型的对象,所以这是一个正式的方法参数。

使用类型参数(如要点3)需要声明类型参数。Java语法是将<T>放在函数前面。这完全类似于在使用方法体中的名称之前向方法声明形式参数名称。

关于List<Object>不接受List<String>,这是有意义的,因为String不是Object;它是Object的子类。修复方法是声明公共静态无效测试(List<?扩展对象> set) ....但是extends Object是多余的,因为每个类都直接或间接地扩展Object。

最后一部分: 虽然String是Object的子集,但是List<String>并不是继承自List<Object>。