当前位置: 首页 > news >正文

泛型(Generic) <? extends T>,<? super T>

通配符边界引入背景

使用泛型的过程中,经常出现一种很别扭的情况。我们有 Fruit 类,和它的派生类 Apple 类。

class Fruit {}
class Apple extends Fruit {}

然后有一个最简单的容器:Plate 类。盘子里可以放一个泛型的 “东西”.

class Plate<T> {private T item;public Plate(T t) {item = t;}public void set(T t) {item = t;}public T get() {return item;}}
代码我们想要表达的意思实际效果
Plate<Apple> p2 = new Plate<Apple> (new Apple());定义一个"苹果盘子" 装 苹果
Plate<Fruit> p = new Plate<Apple> (new Apple());定义一个"水果盘子",逻辑上水果盘子当然可以装苹果error: incompatible types: Plate<Apple> cannot be converted to Plate<Fruit>

实际上,编译器脑袋里认定的逻辑是这样的:

认可关系不认可关系
苹果 IS-A 水果装苹果的盘子 NOT-IS-A 装水果的盘子

就算容器里装的东西之间有继承关系,但容器之间是没有继承关系的。所以我们不可以把Plate<Apple>的引用传递给Plate<Fruit>

为了解决这种情况,Sun的大脑袋们就想出了<? extends T><? super T>的办法,来让”水果盘子“和”苹果盘子“之间发生关系。


<? extends T>上界通配符(Upper Bounds Wildcards)包括T在内的任何T的子类

意义

Plate<Apple>Plate<?extends Fruit>
一个能放苹果的盘子一个能放任意水果的盘子
Plate<?extends Fruit> p = new Plate<Apple>(new Apple());
可以用“苹果盘子”给“水果盘子”赋值了
Plate<?extends Fruit>Plate<Fruit>以及Plate<Apple>的基类

练习

level1level2level3level4
class Food{}class Fruit extends Food{}class Apple extends Fruit{}class RedApple extends Apple{}
class GreenApple extends Apple{}
class Banana extends Fruit{}
class Meat extends Food{}class Pork extends Meat{}
class Beef extends Meat{}

Plate<? extends Fruit> 覆盖区域

level2level3level4
class Fruit extends Food{}class Apple extends Fruit{}class RedApple extends Apple{}
class GreenApple extends Apple{}
class Banana extends Fruit{}

<?superT>下界通配符(Lower Bounds Wildcards)☞包括T在内的任何T的父类

意义

Plate<Fruit>Plate<?super Fruit>
一个能放水果以及一切水果基类的盘子
Plate<?super Fruit>Plate<Fruit>的基类,但不是Plate<Apple>的基类

练习

Plate<?super Fruit>覆盖范围

level1level2
class Food{}class Fruit extends Food{}

上下界通配符的副作用

边界让Java不同泛型之间的转换更容易了。这样的转换也有一定的副作用。那就是容器的部分功能失效
还是以刚才的 Plate 为例。我们可以对盘子做两件事,往盘子里set()新东西,以及从盘子里get()东西。

  1. 上界<? extends T>不能往里存,只能往外取
Plate<? extends Fruit> p=new Plate<Apple>(new Apple());   //不能存入任何元素
p.set(new Fruit());    //Error
p.set(new Apple());    //ErrorFruit newFruit1=p.get();    //读取出来的东西只能存放在Fruit或它的基类里
Object newFruit2=p.get();Apple newFruit3=p.get();    //Error

原因:
编译器只知道容器内是Fruit或者它的派生类,但具体是什么类型不知道。可能是 Fruit 可能是 Apple 也可能是 BananaRedAppleGreenApple

编译器在看到后面用 Plate 赋值以后,盘子里没有被标上有 “苹果”。而是标上一个占位符:CAP#1,来表示捕获一个 FruitFruit的子类,具体是什么类不知道,代号 CAP#1。然后无论是想往里插入 Apple 或者 Meat 或者 Fruit ,译器都不知道能不能和这个 CAP#1 匹配,所以就都不允许。

  1. 下界<? super Fruit>不能往外取,只能往里存
Plate<? super Fruit> p=new Plate<Fruit>(new Fruit());//存入元素正常
p.set(new Fruit());
p.set(new Apple());Object newFruit2=p.get();   //读取出来的东西只能存放在Object类里Apple newFruit3=p.get();    //Error
Fruit newFruit1=p.get();    //Error

原因:
定义的元素是Fruit的基类,那往里存 Fruit及其父类 都可以。但往外读取元素就费劲了,只有 所有类的基类 Object对象 才能装下。但这样的话,元素的类型信息就全部丢失。

http://www.lryc.cn/news/68653.html

相关文章:

  • 数云融合|数字化转型中的利器:揭秘云技术的重要角色
  • Linux篇2
  • 《微服务实战》 第九章 Gitlab使用
  • KMP匹配算法
  • ClickHouse笔记: Ubuntu/Centos下的安装, 配置和用户管理
  • 网络编程——UDP编程
  • linux内核篇-进程及其调度
  • C#开发的OpenRA游戏之基地工程车执行部署命令
  • 米哈游的春招实习面经,问的很基础
  • pro如何添加定时任务
  • bgp路由策略
  • chatGPT4.0编写性能测试报告
  • jpa多线程事务
  • 加密解密软件VMProtect教程(四):准备项目之SDK功能
  • 夏令营教育小程序开发功能和优势有哪些?
  • Cocos CreatorXR 1.2.0 今日发布,正式支持 WebXR ,并开启 MR 之路
  • Linux 使用笔记(本人出品,必属精品)
  • 【2023 · CANN训练营第一季】初识新一代开发者套件 Atlas 200I DK A2 第二章——安装Atlas 200I DK A2跑通第一个案例
  • concurrenthashmap
  • 8年测试总结,项目/团队如何做自动化测试?效率价值?吐血整理...
  • 图像动态裁剪
  • Thematica: 炫彩主题与黑暗奇观的Vue3之旅
  • 平凡的Python为什么能一跃成为世界排名第一的语言
  • Wijmo 2023 v1 Crack
  • 万物互联时代的边缘计算安全需求与挑战
  • 函数序列与函数项级数
  • UML时序图详解
  • Centos7.6部署postgresql15主从
  • 【ThinkPHP6系列学习-2】多应用模式配置
  • Linux内核oops panic简析