深度解析 Java 泛型通配符 `<? super T>` 和 `<? extends T>`

news2025/7/14 10:48:48

Java 泛型中的通配符 ?superextends 关键字组合形成的 <? super T><? extends T> 是泛型系统中最重要的概念之一,也是许多开发者感到困惑的地方。本文将全面剖析它们的语义、使用场景和设计原理。

一、基础概念回顾

1. 泛型通配符 ?

? 表示"未知类型",是泛型系统中的通配符。它解决了泛型不变性(invariance)带来的限制,为泛型系统增加了灵活性。

2. 上下界通配符

  • <? extends T>: 上界通配符(Upper Bounded Wildcard)
  • <? super T>: 下界通配符(Lower Bounded Wildcard)

二、<? extends T> 深入解析

1. 语义含义

表示"某种未知类型,但它是 T 或其子类"。例如:

List<? extends Number> list = new ArrayList<Integer>();

2. 特点

  • 读取安全:可以安全地从集合中读取元素为 T 类型
  • 写入限制:不能向集合中添加任何元素(null 除外)

3. 类型系统原理

<? extends T> 使集合变为生产者(Producer),遵循PECS原则(Producer-Extends)。

public static double sum(List<? extends Number> list) {
    double sum = 0;
    for (Number n : list) {
        sum += n.doubleValue();
    }
    return sum;
}

4. 使用场景

  • 只读取不修改的集合参数
  • 返回不可变视图
  • 实现协变(Covariant)行为

三、<? super T> 深入解析

1. 语义含义

表示"某种未知类型,但它是 T 或其父类"。例如:

List<? super Integer> list = new ArrayList<Number>();

2. 特点

  • 写入安全:可以安全地向集合添加 T 及其子类元素
  • 读取限制:读取的元素只能作为 Object 处理

3. 类型系统原理

<? super T> 使集合变为消费者(Consumer),遵循PECS原则(Consumer-Super)。

public static void addNumbers(List<? super Integer> list) {
    for (int i = 1; i <= 10; i++) {
        list.add(i);
    }
}

4. 使用场景

  • 只写入不读取的集合参数
  • 实现逆变(Contravariant)行为
  • 回调接口设计

四、对比分析

特性<? extends T><? super T>
方向上界(协变)下界(逆变)
读取安全(作为T类型)不安全(只能作为Object)
写入不安全(除null)安全(可添加T及其子类)
PECS角色ProducerConsumer
典型应用数据提供源数据消费端

五、类型系统理论基础

1. 里氏替换原则(LSP)

<? extends T><? super T> 的设计遵循了LSP原则:

  • 子类可以替换父类而不影响程序行为
  • 父类可以接受子类作为参数

2. 协变与逆变

  • 协变(Covariant): 子类型关系与泛型类型关系一致(<? extends T>)
  • 逆变(Contravariant): 子类型关系与泛型类型关系相反(<? super T>)
  • 不变(Invariant): 无子类型关系(普通泛型<T>)

六、高级应用模式

1. PECS 原则

Producer-Extends, Consumer-Super 的缩写,指导通配符使用的黄金法则。

public static <T> void copy(
    List<? super T> dest,  // 消费者,使用super
    List<? extends T> src  // 生产者,使用extends
) {
    for (int i = 0; i < src.size(); i++) {
        dest.add(src.get(i));
    }
}

2. 类型安全的异构容器

结合通配符实现灵活的类型安全容器:

class Favorites {
    private Map<Class<?>, Object> favorites = new HashMap<>();
    
    public <T> void putFavorite(Class<T> type, T instance) {
        favorites.put(Objects.requireNonNull(type), instance);
    }
    
    public <T> T getFavorite(Class<T> type) {
        return type.cast(favorites.get(type));
    }
}

七、常见误区与陷阱

  1. 混淆通配符与类型参数

    // 错误理解
    List<? extends Number> list = new ArrayList<? extends Number>();
    
    // 正确用法
    List<? extends Number> list = new ArrayList<Integer>();
    
  2. 忽略通配符捕获

    // 编译错误
    void swap(List<?> list) {
        Object temp = list.get(0);
        list.set(0, list.get(1));  // 错误
        list.set(1, temp);         // 错误
    }
    
    // 正确方式:使用辅助方法捕获通配符
    void swap(List<?> list) {
        swapHelper(list);
    }
    
    private <E> void swapHelper(List<E> list) {
        E temp = list.get(0);
        list.set(0, list.get(1));
        list.set(1, temp);
    }
    

八、最佳实践建议

  1. 优先使用最严格的类型限制:能用<T>就不用<?>
  2. 遵循PECS原则:明确参数是生产者还是消费者
  3. 避免过度使用通配符:会增加代码复杂度
  4. 合理使用类型推断:结合var关键字简化代码
  5. 文档化类型约束:使用@param说明类型要求

九、实际案例分析

Java集合框架中的应用

// java.util.Collections
public static <T> void copy(
    List<? super T> dest, 
    List<? extends T> src
) {
    // 实现细节
}

// java.util.stream.Stream
<R> R collect(
    Supplier<R> supplier,
    BiConsumer<R, ? super T> accumulator,
    BiConsumer<R, R> combiner
);

十、总结

<? super T><? extends T> 是Java泛型系统的核心特性,它们:

  1. 通过界定类型边界增加了泛型的灵活性
  2. 遵循里氏替换原则和PECS原则
  3. 分别支持协变和逆变行为
  4. 需要开发者深入理解类型系统才能正确使用

掌握这些概念能够帮助开发者设计出更灵活、更类型安全的API,同时也能更好地理解和使用Java集合框架和流API中的高级特性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2343423.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

hbuilderx云打包生成的ipa文件如何上架

使用hbuilderx打包&#xff0c;会遇到一个问题。开发的ios应用&#xff0c;需要上架到app store&#xff0c;因此&#xff0c;就需要APP store的签名证书&#xff0c;并且还需要一个像xcode那样的工具来上架app store。 我们这篇文章说明下&#xff0c;如何在windows电脑&…

Golang | 位运算

位运算比常规运算快&#xff0c;常用于搜索引擎的筛选功能。例如&#xff0c;数字除以二等价于向右移位&#xff0c;位移运算比除法快。

产品动态|千眼狼sCMOS科学相机捕获单分子荧光信号

单分子荧光成像技术&#xff0c;作为生物分子动态研究的关键工具&#xff0c;对捕捉微弱信号要求严苛。传统EMCCD相机因成本高昂&#xff0c;动态范围有限&#xff0c;满阱容量低等问题&#xff0c;制约单分子研究成果产出效率。 千眼狼精准把握科研需求与趋势&#xff0c;自研…

Hot100方法及易错点总结2

本文旨在记录做hot100时遇到的问题及易错点 五、234.回文链表141.环形链表 六、142. 环形链表II21.合并两个有序链表2.两数相加19.删除链表的倒数第n个节点 七、24.两两交换链表中的节点25.K个一组翻转链表(坑点很多&#xff0c;必须多做几遍)138.随机链表的复制148.排序链表 N…

网络:手写HTTP

目录 一、HTTP是应用层协议 二、HTTP服务器 三、HTTP服务 认识请求中的uri HTTP支持默认首页 响应 功能完善 套接字复用 一、HTTP是应用层协议 HTTP下层是TCP协议&#xff0c;站在TCP的角度看&#xff0c;要提供的服务是HTTP服务。 这是在原来实现网络版计算器时&am…

【计算机视觉】CV实战项目 - 基于YOLOv5的人脸检测与关键点定位系统深度解析

基于YOLOv5的人脸检测与关键点定位系统深度解析 1. 技术背景与项目意义传统方案的局限性YOLOv5多任务方案的优势 2. 核心算法原理网络架构改进关键点回归分支损失函数设计 3. 实战指南&#xff1a;从环境搭建到模型应用环境配置数据准备数据格式要求数据目录结构 模型训练配置文…

【python】如何将python程序封装为cpython的库

python程序在发布时&#xff0c;往往会打包为cpython的库&#xff0c;并且根据应用服务器的不同架构&#xff08;x86/aarch64&#xff09;&#xff0c;以及python的不同版本&#xff0c;封装的输出类型也是非常多。本文介绍不同架构指定python下的代码打包方式&#xff1a; 首…

计算机组成原理 课后练习

例一&#xff1a; 例二&#xff1a; 1. 原码一位乘 基本原理 原码是一种直接表示数值符号和大小的方式&#xff1a;最高位为符号位&#xff08;0表示正&#xff0c;1表示负&#xff09;&#xff0c;其余位表示数值的绝对值。原码一位乘的核心思想是逐位相乘&#xff0c;并通…

SVN仓库突然没有权限访问

如果svn仓库突然出现无法访问的情况&#xff0c;提示没有权限&#xff0c;所有账号都是如此&#xff0c;新创建的账号也不行。 并且会突然提示要输入账号密码。 出现这个情况时&#xff0c;大概率库里面的文件有http或者https的字样&#xff0c;因为单独给该文件添加权限导致…

【Qt】文件

&#x1f308; 个人主页&#xff1a;Zfox_ &#x1f525; 系列专栏&#xff1a;Qt 目录 一&#xff1a;&#x1f525; Qt 文件概述 二&#xff1a;&#x1f525; 输入输出设备类 三&#xff1a;&#x1f525; 文件读写类 四&#xff1a;&#x1f525; 文件和目录信息类 五&…

【AI】[特殊字符]生产规模的向量数据库 Pinecone 使用指南

一、Pinecone 的介绍 Pinecone是一个完全托管的向量数据库服务&#xff0c;专为大规模机器学习应用设计。它允许开发者轻松存储、搜索和管理高维向量数据&#xff0c;为推荐系统、语义搜索、异常检测等应用提供强大的基础设施支持。 1.1 Pinecone的核心特性 1. 高性能向量搜…

dstream

DStream转换DStream 上的操作与 RDD 的类似&#xff0c;分为 Transformations&#xff08;转换&#xff09;和 Output Operations&#xff08;输出&#xff09;两种&#xff0c;此外转换操作中还有一些比较特殊的原语&#xff0c;如&#xff1a;updateStateByKey()、transform(…

HFSS5(李明洋)——设置激励(波端口激励)

Magnetic是适用于铁磁氧导体的&#xff0c;只有前三种激励类型可以用于计算S参数 1波端口激励 也可以设置在模型内部&#xff0c;如果是设置在模型内部必须加一段理想导体&#xff0c;用于指定端口方向 1.1——模式 number 输入N&#xff1a;计算1-N的模式都计算 1.2——模式…

ubiquant比赛系列——用docker准备ubipoker开发环境

比赛过程&#xff1a; 环境准备&#xff1a; #在云服务器上拉python官方的docker镜像并下载到本地 https://hub.docker.com/_/python/ sudo docker pull python:3.11.12-slim-bullseye sudo docker images sudo docker save -o 3.11.12-slim-bullseye.tar python:3.11.12-slim…

Rust实现高性能目录扫描工具ll的技术解析

Rust实现高性能目录扫描工具ll的技术解析 一、项目概述 本项目使用Rust构建了一个类ls命令行工具&#xff0c;具备以下核心特性&#xff1a; 多格式文件信息展示并行目录扫描加速人类可读文件大小运行时性能统计交互式进度提示 二、技术架构 1. 关键技术栈 clap&#xff…

深入理解C语言变量:从基础到实践

在编程世界中&#xff0c;变量是最基础也是最重要的概念之一。作为C语言的核心组成部分&#xff0c;变量承载着程序运行时数据的存储和传递功能。理解变量的工作原理和正确使用方法&#xff0c;是成为一名合格C程序员的必经之路。本文将全面介绍C语言变量的各个方面&#xff0c…

RK3562/3588 系列之0—NPU基础概念

RK3562/3588 系列之0—NPU基础概念 1.处理器分类2.算力衡量指标TOPS参考文献 1.处理器分类 中央处理器(CPU)&#xff1b; 图形处理器 (GPU)&#xff1b; 神经网络处理器(NPU)。 每个处理器擅长不同的任务:CPU擅长顺序控制和即时性&#xff1b;GPU适合并行数据流处理,NPU擅长…

canvas画板!随意画!!

希望你天天开心 代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>绘画板</title…

Float32、Float16、BFloat16

我们先介绍 Float32、Float16、BFloat16 的 浮点数表示方法 然后根据浮点数表示&#xff0c;来分析总结他们是怎么控制 精度和 数值范围 的 最后再来对比的说明 Float32、Float16、BFloat16 的 应用场景 和 硬件支持 1、浮点数的表示方法 Float32 &#xff1a; 单精度浮点数…

5V 1A充电标准的由来与技术演进——从USB诞生到智能手机时代的电力革命

点击下面图片带您领略全新的嵌入式学习路线 &#x1f525;爆款热榜 88万阅读 1.6万收藏 一、起源&#xff1a;USB标准与早期电力传输需求 1. USB的诞生背景 1996年&#xff0c;由英特尔、微软、IBM等公司组成的USB-IF&#xff08;USB Implementers Forum&#xff09;发布了…