List<? extends T>与List<? super T>的区别

ps:标题中的圆括号应该为尖括号,否则,标题非法,发表不出去。


一、前置名次解释

(1)?

?表示类型通配符,即具体传什么参数类型,在List定义时不用考虑。

(2)<T>

这里的<>表示泛型,T表示泛型中装载的类型为T类型,等到需要的时候,我们可以具体这个T。我们在使用动态数组实现ArrayList的时候,如果希望这个ArrayList不仅仅支持一个类型的话,我们可以给这个ArrayList定义泛型,泛型中存放的是T类型。在实际创建出这个ArrayList对象时,我们可以指定泛型中的具体类型。

(3)<? extends T>

类型上界。表示的是此处我们传入的类型必须是T类型本身或其子类。如果这里的T为Number的话,那么我们只能传入Number本身或其子类,比如Integer,Floart等基本数据类型的包装类。

(4)<? super T>

类型下界。表示的是此处我们传入的类型必须是T类型本身或其超类。如果这里的T为Integer的话,那么我们只能传入Integer本身或其超类,比如Number类。


二、List<? extends T>

<? extends T>是被设计用来读取数据的泛型,并且只能读取类型为T的元素。原因如下:

我们现在知道,此List中存放的都是T类型或其子类,因此,我们可以这样做来读取元素。

   List<? extends Number> list = new ArrayList<>(); 
   Number number=list.get(0);

使用Number类型,即T类型来接受本身类型或其子类类型的元素。

当然这样的赋值也是合法的

   List<? extends Number> integers = new ArrayList<Integer>();
   List<? extends Number> floats = new ArrayList<Float>();

他为什么不能增加元素?原因如下:

编译器仅仅通过List<? extends Number>来了解到,这个List中可以存放Number本身和其子类,但具体是什么它不知道,即使在等号后面使用new ArrayList<Integer>(),编译器仍然不为所动,它就是不知道该存什么类型,那么,这种泛型,就不能用来临时add元素。

但是!虽然不能add元素,但可以在初始化的时候,接受一个已经定义好的list,而该list存放的类型一定相同,因此,List<? extends T>可直接接受一个定义好的list。

 public static List<Integer> getList(){
        List<Integer> list=new ArrayList<>();
        list.add(1);
        return list;
    }

....

    public static void main(String[] args) {
        List<? extends Number> list = new ArrayList<>();
        list=getList();
    }

三、List<? super T>

List<? super T>是被设计用来添加数据的泛型,并且只能添加T类型或其子类类型的元素。

我们知道,此List中存放的元素一定是T类型或其父类类型的元素,由于编译器不确定到底是什么类型,我们可以这样来添加元素。

  List<? super Integer> list2 = new ArrayList<>();
        list2.add(new Integer(1));
        list2.add(new Double(1.0));//编译错误,Double不为Integer的子类

由于编译器不知道我们需要什么类型,因此在获取元素的时候,总是返回Object类型,因此我们使用Object类型来接受获取到的元素。

   List<? super Integer> list2 = new ArrayList<>();
   list2.add(new Integer(1));
   Object integer=list2.get(0);

如果我们非要使用List<? super Integer>中的Integer类型来接收获取到的元素,那么必须进行强制类型转换。

   List<? super Integer> list2 = new ArrayList<>();
   list2.add(new Integer(1));
   Integer integer1= (Integer) list2.get(0);

如果我们使用Object类型来接收获取到的元素,那么元素本身的类型就会丢失,因此,我们不使用List<? super T>来获取元素。


四、总结

(1)List<? extends T>适用于读取数据,读取出来的数据全部用T类型接收。如果我们往此list中添加T类型不同的子类的话,各种子类无法相互转换,因此不能添加元素,但可接受初始赋值。

(2)List<? super T>适用于添加元素,只能添加T类型或其子类类型。因为这些类型都能转换为?表示的类型(向上转型),因此我们可以对此list添加元素。只能用Object类型来接收获取到的元素,但是这些元素原本的类型会丢失,因此最好不要使用此泛型来获取元素。