Java泛型中extends和super的区别?

本文详细介绍了Java中泛型通配符`extends`和`super`的用法。`extends`用于读取操作,确保能读取到指定类型或其子类的对象,但不允许插入;`super`则用于写入操作,允许添加指定类型或其超类的对象,但读取时类型不确定。理解PECS原则(Producer Extends, Consumer Super)有助于更好地运用这两种通配符。
摘要由CSDN通过智能技术生成
1.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,因为foo3可能指向的是List

  • 写入操作过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?

你不能插入一个Integer元素,因为foo3可能指向List

你不能插入一个Double元素,因为foo3可能指向List

你不能插入一个Number元素,因为foo3可能指向List

你不能往List<? extends T>中插入任何类型的对象,因为你不能保证列表实际指向的类型是什么,你并不能保证列表中实际存储什么类型的对象。唯一可以保证的是,你可以从中读取 到T或者T的子类。

2.super

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或者List

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

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

  • 写入操作通过以上给定的赋值语句,你能把一个什么类型的元素合法地插入到foo3中呢?
  • 你可以插入Integer对象,因为上述声明的列表都支持Integer

你可以插入Integer的子类的对象,因为Integer的子类同时也是Integer,原因同上。

你不能插入Double对象,因为foo3可能指向ArrayList

你不能插入Number对象,因为foo3可能指向ArrayList

你不能插入Object对象,因为foo3可能指向ArrayList

3.PECS

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

 

  • 生产者使用extends
  • 如果你需要一个列表提供T类型的元素(即你想从列表中读取T类型的元素),你需要把这个列表声明成<? extends T>,比如List<? extends Integer>,因此你不能往该列表中添加任何元素。
  • 消费者使用super
  • 如果需要一个列表使用T类型的元素(即你想把T类型的元素加入到列表中),你需要把这个列表 声明成<? super T>,比如List<? super Integer>,因此你不能保证从中读取到的元素的类型。
  • 即是生产者,也是消费者
  • 如果一个列表即要生产,又要消费,你不能使用泛型通配符声明列表,比如List

2.上下界通配符的副作用

 

  • 边界让Java不同泛型之间的转换更容易了。但不要忘记,这样的转换也有一定的副作用。那就是容 器的部分功能可能失效。
  • 上界<? extends T>不能往里存,只能往外取。
  • 下界<? super T>不影响往里存,但往外取只能放在Object对象里
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值