Java 泛型:上界通配符和下界通配符的用途和限制

Java 泛型:上界通配符和下界通配符的用途和限制

Java泛型中的上界通配符(<? extends T>)和下界通配符(<? super T>)是两个强大的工具,它们在不同的场景中各自发挥重要作用。理解它们的用途和限制对于编写类型安全且高效的泛型代码至关重要。本文将详细解释为什么上界通配符通常用于只需要读取数据的场景,而下界通配符通常用于只需要写入数据的场景。

1. 上界通配符(<? extends T>

用于只需要读取数据的场景

上界通配符 <? extends T> 表示类型必须是 TT 的子类。在这种情况下,你只能安全地从集合中读取数据,而不能往集合中写入数据。原因如下:

例如,如果 AnimalCreature 的子类,而 Cat 又是 Animal 的子类,那么 List<? extends Creature> 可以是 List<Animal>List<Cat>

java 复制代码
class Creature {}
class Animal extends Creature {}
class Cat extends Animal {}

public void method(List<? extends Creature> list) {
    // 可以读取
    for (Creature creature : list) {
        System.out.println(creature);
    }
    // 不可以写入
    list.add(new Animal()); // 编译错误
    list.add(new Cat()); // 仍然编译错误
}

在这个例子中,List<? extends Creature> 可以是 List<Animal>List<Cat> 等。我们只能读取其中的 Creature 对象,而不能往里面添加任何元素(即使是 Creature 类型的元素),否则会引发编译错误。使用反证法,如果 method 的实参是 new ArrayList<Cat>(),而 method 的形参是可以接受泛型为 Animal 的,那么在方法体中给 List<Cat> 添加 Animal 对象显然是不合适的。因此,编译器禁止了这种写入操作。

2. 下界通配符(<? super T>

用于只需要写入数据的场景

下界通配符 <? super T> 表示类型必须是 TT 的父类。在这种情况下,你只能安全地往集合中写入数据,而不能从集合中读取特定类型的数据。

反证法解释

假设 List<? super Animal> 可以读取为 Animal 类型:

java 复制代码
public void method(List<? super Animal> list) {
    Animal animal = list.get(0); // 假设可以读取为 Animal 类型
}

假如 method 方法接受一个 List<Creature> 作为参数:

java 复制代码
List<Creature> creatures = new ArrayList<>();
method(creatures);

在方法内部,如果我们尝试将 list.get(0) 读取为 Animal 类型,就会引发类型错误,因为 List<Creature> 可能包含非 AnimalCreature 对象。

这种假设会导致类型安全问题,因此编译器禁止从 List<? super Animal> 中读取为 Animal 类型。只能安全地向其中添加 Animal 或其子类对象。

读取数据的限制

只能读取 Object 类型:

java 复制代码
public void readObjects(List<? super Cat> list) {
    Object obj = list.get(0); // 可以读取,但类型是 Object
}

下界通配符确保了类型安全,避免了类型不一致的问题。

总结

  • 上界通配符(<? extends T> :用于只需要读取数据的场景,因为它允许使用 T 的任何子类,但禁止写入数据以保证类型安全。
  • 下界通配符(<? super T> :用于只需要写入数据的场景,因为它允许向集合中添加 T 及其子类,但读取的数据只能保证是 Object 类型。
  • 上吐下泻有上限只能读(吐),有下限只能写(泻)

希望这篇文章能帮助你更好地理解和应用Java泛型中的上界通配符和下界通配符。如果你有任何问题或建议,欢迎在评论区留言!

相关推荐
互联网全栈架构23 分钟前
遨游Spring AI:第一盘菜Hello World
java·人工智能·后端·spring
优秀的颜1 小时前
计算机基础知识(第五篇)
java·开发语言·分布式
BillKu1 小时前
Java严格模式withResolverStyle解析日期错误及解决方案
java
网安INF1 小时前
ElGamal加密算法:离散对数难题的安全基石
java·网络安全·密码学
AWS官方合作商2 小时前
在CSDN发布AWS Proton解决方案:实现云原生应用的标准化部署
java·云原生·aws
gadiaola3 小时前
【JVM】Java虚拟机(二)——垃圾回收
java·jvm
coderSong25686 小时前
Java高级 |【实验八】springboot 使用Websocket
java·spring boot·后端·websocket
Mr_Air_Boy7 小时前
SpringBoot使用dynamic配置多数据源时使用@Transactional事务在非primary的数据源上遇到的问题
java·spring boot·后端
豆沙沙包?7 小时前
2025年- H77-Lc185--45.跳跃游戏II(贪心)--Java版
java·开发语言·游戏
年老体衰按不动键盘8 小时前
快速部署和启动Vue3项目
java·javascript·vue