Java 泛型中super T和extends T的区别

开发 后端
经常发现有List< super T>、Set<extends T>的声明,是什么意思呢?< super T>表示包括T在内的任何T的父类,< extends T>表示包括T在内的任何T的子类,下面我们详细分析一下两种通配符具体的区别。

经常发现有List< super T>、Set<extends T>的声明,是什么意思呢?< super T>表示包括T在内的任何T的父类,< extends T>表示包括T在内的任何T的子类,下面我们详细分析一下两种通配符具体的区别。

extends

  1. List< extends Number> foo3的通配符声明,意味着以下的赋值是合法的: 
  2.  
  3. // Number "extends" Number (in this context) 
  4.  
  5. List< extends Number> foo3 = new ArrayList< extends Number>(); 
  6.  
  7. // Integer extends Number 
  8.  
  9. List< extends Number> foo3 = new ArrayList< extends Integer>(); 
  10.  
  11. // Double extends Number 
  12.  
  13. List< extends Number> foo3 = new ArrayList< extends Double>(); 
  • 读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?你可以读取到Number,因为以上的列表要么包含 Number元素,要么包含Number的类元素。你不能保证读取到Integer,因为foo3可能指向的是List<Double>。你 不能保证读取到Double,因为foo3可能指向的是List<Integer>。

  • 写入操作过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?你不能插入一个Integer元素,因为foo3可能指向 List<Double>。你不能插入一个Double元素,因为foo3可能指向List<Integer>。你不能插入一个 Number元素,因为foo3可能指向List<Integer>。你不能往List< extends T>中插入任何类型的对象,因为你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。***可以保证的是,你可以从中读 取到T或者T的子类。

super

现在考虑一下List< super T>。

 

  1. List< super Integer> foo3的通配符声明,意味着以下赋值是合法的: 
  2.  
  3. // Integer is a "superclass" of Integer (in this context) 
  4.  
  5. List< super Integer> foo3 = new ArrayList<Integer>(); 
  6.  
  7. // Number is a superclass of Integer 
  8.  
  9. List< super Integer> foo3 = new ArrayList<Number>(); 
  10.  
  11. // Object is a superclass of Integer 
  12.  
  13. List< super Integer> foo3 = new ArrayList<Object>(); 
  • 读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?你不能保证读取到Integer,因为foo3可能指向 List<Number>或者List<Object>。你不能保证读取到Number,因为foo3可能指向 List<Object>。***可以保证的是,你可以读取到Object或者Object子类的对象(你并不知道具体的子类是什么)。

  • 写入操作通过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?你可以插入Integer对象,因为上述声明的列表都支持 Integer。你可以插入Integer的子类的对象,因为Integer的子类同时也是Integer,原因同上。你不能插入Double对象,因为 foo3可能指向ArrayList<Integer>。你不能插入Number对象,因为foo3可能指向 ArrayList<Integer>。你不能插入Object对象,因为foo3可能指向 ArrayList<Integer>。

PECS

请记住PECS原则:生产者(Producer)使用extends,消费者(Consumer)使用super。

生产者使用extends

如果你需要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你需要把这个列表声明成< extends T>,比如List< extends Integer>,因此你不能往该列表中添加任何元素。

消费者使用super

如果需要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你需要把这个列表声明成< super T>,比如List< super Integer>,因此你不能保证从中读取到的元素的类型。

即是生产者,也是消费者

如果一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,比如List<Integer>。

例子

请参考java.util.Collections里的copy方法(JDK1.7):

泛型中 super T和 extends T的区别

我们可以从Java开发团队的代码中获得到一些启发,copy方法中使用到了PECS原则,实现了对参数的保护。

 
责任编辑:王雪燕 来源: 并发编程网
相关推荐

2019-12-12 10:25:33

Java泛型编程语言

2012-02-16 10:36:43

Java

2021-07-27 07:31:16

JavaArrayList数组

2023-10-07 00:01:02

Java函数

2009-03-17 16:22:13

Java泛型接口

2009-09-25 10:03:51

Java泛型

2021-06-17 06:51:32

Java泛型Java编程

2024-01-24 08:31:13

extends​接口规范

2017-03-06 16:51:52

Java泛型实现

2023-12-15 10:11:31

数据结构方式

2011-07-08 13:40:18

2021-09-29 18:17:30

Go泛型语言

2024-04-23 08:23:36

TypeScript泛型Generics

2021-12-30 19:34:15

Java泛型JDK

2020-10-26 14:01:22

Java泛型

2009-06-16 11:32:00

Java泛型

2009-09-14 18:57:19

LINQ查询

2011-06-03 08:49:54

Java

2009-10-10 09:27:42

Java泛型通用方法

2009-06-11 17:31:27

Java泛型
点赞
收藏

51CTO技术栈公众号