原文链接          李璟(jlee381344197@gmail.com)

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

extends

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

01 // Number "extends" Number (in this context)
02  
03 List<? extends Number> foo3 = new ArrayList<? extends Number>();
04  
05 // Integer extends Number
06  
07 List<? extends Number> foo3 = new ArrayList<? extends Integer>();
08  
09 // Double extends Number
10  
11 List<? extends Number> foo3 = new ArrayList<? extends Double>();
  1. 读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?你可以读取到Number,因为以上的列表要么包含Number元素,要么包含Number的类元素。

    你不能保证读取到Integer,因为foo3可能指向的是List<Double>。

    你不能保证读取到Double,因为foo3可能指向的是List<Integer>。

  2. 写入操作过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到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的通配符声明,意味着以下赋值是合法的:

01 // Integer is a "superclass" of Integer (in this context)
02  
03 List<? super Integer> foo3 = new ArrayList<Integer>();
04  
05 // Number is a superclass of Integer
06  
07 List<? super Integer> foo3 = new ArrayList<Number>();
08  
09 // Object is a superclass of Integer
10  
11 List<? super Integer> foo3 = new ArrayList<Object>();
  1. 读取操作通过以上给定的赋值语句,你一定能从foo3列表中读取到的元素的类型是什么呢?你不能保证读取到Integer,因为foo3可能指向List<Number>或者List<Object>。

    你不能保证读取到Number,因为foo3可能指向List<Object>。

    唯一可以保证的是,你可以读取到Object或者Object子类的对象(你并不知道具体的子类是什么)。

  2. 写入操作通过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到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原则,实现了对参数的保护。

原创文章,转载请注明: 转载自并发编程网 – ifeve.com本文链接地址: 泛型中? super T和? extends T的区别

Java 泛型中? super T和? extends T的区别相关推荐

  1. 泛型中? super T和? extends T的区别

    经常发现有List<? super T>.Set<? extends T>的声明,是什么意思呢?<? super T>表示包括T在内的任何T的父类,<? ex ...

  2. java面试 泛型_Java面试题五:Java 的泛型, super T 和 extends T 的区别

    相关文章: java泛型通配符 ? extends T 和 ? super T 的区别和用法 一.二者的意思 ? extends T :表示上界是T, ? 都是继承自T的,都是T的子类: ? supe ...

  3. Java泛型中extends T和super T的区别?

    <? extends T>和<? super T>是Java泛型中的"通配符(Wildcards)"和"边界(Bounds)"的概念. ...

  4. Java泛型中extends和super的理解(转)

    E – Element (在集合中使用,因为集合中存放的是元素) T – Type(Java 类) K – Key(键) V – Value(值) N – Number(数值类型) ? – 表示不确定 ...

  5. Java泛型中extends和super的区别?

    区别 <? extends T>和<? super T>是Java泛型中的"通配符(Wildcards)"和"边界(Bounds)"的概 ...

  6. Java泛型中? 和 ? extends Object的异同分析

    点击上方蓝色"程序猿DD",选择"设为星标" 回复"资源"获取独家整理的学习资料! 作者 | 刘一手 来源 | 公众号「锅外的大佬」 Jav ...

  7. 聊一聊Java 泛型中的通配符 T,E,K,V,?

    点击上方"方志朋",选择"设为星标" 回复"1024"获取独家整理的学习资料 作者:glmapper juejin.im/post/5d57 ...

  8. 聊一聊-JAVA 泛型中的通配符 T,E,K,V,?

    前言 Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机制,该机制允许开发者在编译时检测到非法的类型. 泛型的本质是参数化类型,也就是说所操作的数据 ...

  9. Java泛型中的PECS原则

    今天在写代码的时候使用到了这样一个方法签名: public void foo(Map<String, String> map); 在写这个参数的时候正好在想一些关于泛型的东西,于是: pu ...

最新文章

  1. 前端js判断上传是否为EXCEL表格问题
  2. php v5.,PHP V5.3 中的新特性,第 5 部分- 从 PHP V5.2 升级到 PHP V5.3
  3. 如何在Eclipse中开启代码自动提示功能
  4. 学习RadonDB源码(三)
  5. nlp3-有限自动机FLandFA
  6. 魔兽世界联盟物价稳定的服务器,一个残酷的真相!在《魔兽世界》怀旧服,女生一般都爱选联盟...
  7. thymeleaf的url属性
  8. python实现规则引擎_python – 如何在不使用eval()或exec()的情况下创建规则引擎?...
  9. Codeforces Round #753 (Div. 3)E. Robot on the Board 1
  10. NLP︱LDA主题模型的应用难题、使用心得及从多元统计角度剖析
  11. Vmware安装vmware-tools后,仍无法上网
  12. Markdown常用数学符号
  13. 91 全能地图下载器和 bigemap地图下载器哪个好用?
  14. 自适应鲁棒控制(ARC)实例推导(手写超详细)
  15. 码云仓库第一次上传代码流程和git相关操作合集(持续更新)
  16. Java对Word文档进行操作
  17. 法院判错案如何追责?
  18. go和python优缺点_我为什么放弃了 Python ,选择了 Go?
  19. 文件与磁盘空间管理---外存分配方式、存储空间管理
  20. 94.(leaflet之家)leaflet态势标绘-进攻方向绘制(燕尾)

热门文章

  1. 10次相遇我才知道什么是成熟的爱--转
  2. spring mvc DispatcherServlet详解之四---视图渲染过程
  3. 产品分析报告|读书新贵——《网易蜗牛读书》
  4. 【项目实战】P2P金融数据指标分析
  5. The Illustrated Transformer:中英文(看原文,很多翻译是错误的)
  6. 想开网店?向你推荐最好的开源电子商务平台
  7. 物联网产品:你需要知道的9种智能家居产品
  8. Nginx-从零开始使用nginx实现反向代理及负载均衡
  9. Redis-01Redis概述
  10. Java学习笔记(三)--Java主类结构