泛型extends和super的区别

在 Java 泛型中,extends 和 super 是用于限定类型边界的两个关键字。它们的核心区别在于‌限定的方向‌(上界 vs 下界)以及由此决定的‌读写权限‌(Producer vs Consumer)。

理解这一区别的最佳方式是遵循 ‌PECS 原则‌:‌P‌roducer-‌E‌xtends, ‌C‌onsumer-‌S‌uper。

  1. 核心对比总结
  2. 详细解析:<? extends T> (上界)
    ‌定义‌:表示未知的具体类型,但该类型必须是 T 或者 T 的子类。

‌为什么只能读(Producer)?‌

假设你有一个 List<? extends Number>,它可能实际上是 List,也可能是 List。

‌读取安全‌:无论它是 Integer 还是 Double,它们都是 Number 的子类。因此,当你从中取出元素时,你可以安全地将其视为 Number 类型。

‌写入不安全‌:如果你尝试 list.add(1) (Integer),但实际列表是 List,就会破坏类型安全。反之亦然。因为编译器无法确定具体的子类类型,所以禁止添加任何非 null 元素。

代码示例:

java 复制代码
public void processNumbers(List<? extends Number> list) {
    // ✅ 安全读取:所有元素至少是 Number
    for (Number num : list) {
        System.out.println(num.doubleValue());
    }
    
    // ❌ 编译错误:不能确定具体类型,禁止添加
    // list.add(1); 
}
  1. 详细解析:<? super T> (下界)
    ‌定义‌:表示未知的具体类型,但该类型必须是 T 或者 T 的父类。

‌为什么只能写(Consumer)?‌

假设你有一个 List<? super Integer>,它可能实际上是 List,List 或 List。

‌写入安全‌:无论列表的具体类型是什么,它一定是 Integer 的父类(或本身)。因此,向其中添加 Integer 对象永远是安全的(Integer 可以向上转型为 Number 或 Object)。

‌读取受限‌:当你从列表中取出元素时,编译器只知道它是 Integer 的某个父类。可能是 Integer,也可能是 Number,甚至是 Object。为了类型安全,编译器只能保证取出的对象是 Object 类型,无法直接当作 Integer 使用。

‌代码示例‌:

java 复制代码
public void addIntegers(List<? super Integer> list) {
    // ✅ 安全写入:Integer 可以存入 Integer, Number, Object 列表
    list.add(10);
    list.add(20);
    
    // ⚠️ 读取受限:只能当作 Object 读取
    Object obj = list.get(0); 
    // ❌ 编译错误:不能保证取出的就是 Integer
    // Integer num = list.get(0); 
}
  1. PECS 原则实战应用
    PECS 原则帮助你在设计 API 时决定使用哪种通配符:

‌如果你需要从参数中获取数据(生产数据)‌,使用 extends。

例如:Collections.max(Collection<? extends T>) ------ 它需要遍历集合并比较元素,不需要修改集合。

‌如果你需要向参数中放入数据(消费数据)‌,使用 super。

例如:Collections.copy(List<? super T> dest, List<? extends T> src) ------ 目标列表 dest 需要接收来自源列表的数据,所以是消费者,用 super;源列表 src 提供数据,所以是生产者,用 extends。

  1. 常见误区澄清

‌ vs <? extends Number>‌:

用在‌类或方法定义‌中,表示一个确定的但未知的类型 T。在方法内部,你可以既读又写(只要类型一致),因为 T 是固定的。
<? extends Number> 用在‌变量声明或方法参数‌中,表示一个未知的类型范围。它主要用于提高 API 的灵活性,允许传入更多类型的实参,但牺牲了读写能力。 ‌null 的特殊性‌: null 是所有引用类型的成员。因此,无论是 extends 还是 super,都可以向集合中添加 null。 总结口诀 ‌Extends 上界,只读不写‌(因为不知道具体是哪个子类,怕写错)。 ‌Super 下界,只写不读‌(因为不知道具体是哪个父类,读出来只能用 Object)。

相关推荐
小bo波8 小时前
Java Swing 图形用户界面实验 —— 从算术练习到游戏开发的完整实践
java·课程设计·gui·游戏开发·扫雷·swing
咖啡八杯10 小时前
GoF设计模式——备忘录模式
java·后端·spring·设计模式
SamDeepThinking20 小时前
裁掉那个差程序员后,给你看团队里高手的代码:这个习惯,希望你有
java·后端·程序员
朕瞧着你甚好21 小时前
技术雷达 & Java 集成评估报告 — Apache Tika 3.3.1
java·ai编程
MacroZheng1 天前
短短几天,暴涨2.8万Star!又一款编程神器开源!
java·人工智能·后端
SamDeepThinking1 天前
函数式编程:用BiFunction消除多类型分支的代码重复
java·后端·面试
Flittly2 天前
【AgentScope Java新手村系列】(16)从RAG到多路检索
java·spring boot·spring
小兔崽子去哪了2 天前
Java 生成二维码解决方案
java·后端
人活一口气2 天前
从JVM调优到MCP协议:Java全栈技术体系深度总结与企业级架构实践
java·spring boot
NE_STOP2 天前
Vibe Coding -- 完整项目案例实操
java