泛型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)。

相关推荐
weixin_523185321 小时前
Java内存模型详解:栈、堆、方法区、本地方法栈与程序计数器
java·开发语言
惜缘破军1 小时前
基于 Spring Boot 4 和 Spring Cloud 2025 的微服务基础框架 hdfk7-boot
java
小白起 v1 小时前
从零搭建一个现代化的验证码登录系统:Spring Boot + 阿里云短信实战教程
java·阿里云
未若君雅裁2 小时前
工厂模式详解:简单工厂、工厂方法与抽象工厂
java·开发语言
不会写DN2 小时前
通过php 中的Route:: 的写法了解什么是静态类调用
android·java·php
小刘|2 小时前
SpringAIAlibaba快速接入阿里云百炼
java·spring boot·spring·maven
我命由我123452 小时前
由 ImageView 获取到的 Drawable 对象,它的 intrinsicWidth、intrinsicWidth 与实际图片的尺寸
java·开发语言·java-ee·android studio·android jetpack·android-studio·android runtime
Han.miracle2 小时前
Jackson 工具类详解:ObjectMapper 配置、泛型擦除、TypeReference 与 JavaType
java·spring boot·spring
guslegend2 小时前
Java 创建对象有几种方式
java·开发语言