干货丨Java 泛型中super T和extends T的区别

createh53周前 (12-18)技术教程19

代码中经常发现有List< super T>、Set的声明,是什么意思呢?

< super T>表示包括T在内的任何T的父类,< extends T>表示包括T在内的任何T的子类,下面我们详细分析一下两种通配符具体的区别。

extends

List< extends Number> foo3的通配符声明,意味着以下的赋值是合法的:

// Number "extends" Number (in this context)

List< extends Number> foo3 = new ArrayList< extends Number>;

// Integer extends Number

List< extends Number> foo3 = new ArrayList< extends Integer>;

// Double extends Number

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>

List< super Integer> foo3的通配符声明,意味着以下赋值是合法的:

// Integer is a "superclass" of Integer (in this context)

List< super Integer> foo3 = new ArrayList<Integer>;

// Number is a superclass of Integer

List< super Integer> foo3 = new ArrayList<Number>;

// Object is a superclass of Integer

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):

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

(内容来源:并发编程网)

相关文章

2 分钟快速搞懂,Java 泛型中的通配符 T,E,K,V

Java 泛型中的通配符 T , E , K , V , ? 是什么?经常有同学会分不清楚。本文我们一起来了解下。送:《泛型最全知识导图》、《大厂泛型面试真题26道》,到本篇结尾处获得~1 什么是泛型...

PHP 8.0正式发布:支持JIT编译器,性能提升高达3倍

美国时间11月26日,PHP团队宣布PHP 8.0正式GA。PHP 8.0是PHP语言的最新主要版本,带来了许多新特性和优化,包括命名参数(named arguments)、联合类型(union ty...

升级IDEA后Lombok不能用了,如何解决?

今天到工作室比较晚,在电脑前吃着早饭,看到提示IDEA提示升级,寻思已经有好久没有升过级了。一样等着,就升级下吧。升级完毕重启之后,突然发现好多错误,原来的应用也没法启动了。仔细一看报错信息,是由于L...

SpringBoot 2.5.5整合轻量级的分布式日志标记追踪神器TLog

TLog能解决什么痛点随着微服务盛行,很多公司都把系统按照业务边界拆成了很多微服务,在排错查日志的时候。因为业务链路贯穿着很多微服务节点,导致定位某个请求的日志以及上下游业务的日志会变得有些困难。这时...

为什么有很多人跑去日本当程序员?

看到这个问题的一瞬间,立马就想到我一学土木的朋友,他现在是在设计院一个月三四千的工资卖命,经常跟我说,他本科同专业的同学毕业自学编程,再加上因为沉迷二次元喜欢日语,日语水平也不错,现在找了个工作被外派...