上次跟你们聊了泛型的基础用法,今天接着往下说 —— 泛型里还有个挺重要的概念叫 “通配符”,就是那个问号 “?”,很多人第一次见都懵:这玩意儿跟普通泛型有啥区别?为啥有时候非得用它不可?小索奇当初也卡这儿好久,后来拿实际例子一琢磨,才彻底明白它的用处。

先给你出个题:假如你写了个方法,要打印所有 List 集合里的元素,不管这个 List 装的是 String、Integer 还是别的类型,该咋写?

你可能会想,直接用 List不就行了?比如这样:

public static void printList(List list) {

for (T item : list) {

System.out.println(item);

}

}

这么写没问题,但还有个更简洁的方式 —— 用通配符 “?”:

public static void printList (List<?> list) {

for (Object item : list) {

System.out.println (item);

}

}

看出区别没?用通配符的话,不用定义泛型参数,代码更短。而且这两种写法效果一样,不管你传 List还是 List,都能正常打印。那为啥要有两种写法呢?其实核心区别在于:如果方法里不需要用到泛型的具体类型,只用通配符就够了;要是需要操作具体类型(比如给集合加元素),就得用

比如你想写个方法,给 List 里加一个默认元素,这时候用通配符就不行了:

// 这段代码会编译报错!

public static void addDefault (List<?> list) {

list.add ("默认值"); // 报错:无法确定?的类型,不能加 String

}

这背后的原因是啥呢?因为通配符 “?” 代表 “未知类型”,编译器不知道这个 List 到底装的是啥,自然不敢让你随便加元素 —— 万一人家是 List,你硬塞个 String 进去,不就乱套了?

但用就能搞定:

public static void addDefault(List list, T defaultValue) {

list.add(defaultValue);

}

调用的时候指定类型就行,比如给 List加默认值:

List strList = new ArrayList<>();

addDefault (strList, "默认值"); // 没问题

是不是一下子就懂了?简单说就是:通配符适合 “只读” 场景(比如打印集合),泛型参数适合 “读写” 场景(比如给集合加元素)。

除了单纯的 “?”,通配符还有两种常用写法:<? extends T > 和 <? super T>,这俩也特容易搞混,小索奇当初记了好几天才分清。

先看 <? extends T>,它代表 “T 及其子类”。比如 <? extends Number>,就包括 Integer、Double、Long 这些 Number 的子类。这种写法的特点是 “能读不能写”(跟普通?类似,但多了类型限制)。比如你写个方法,求所有 Number 类型集合的总和:

public static double sum (List<? extends Number> list) {

double total = 0;

for (Number num : list) {

total += num.doubleValue ();

}

return total;

}

不管你传 List还是 List,都能算总和,因为它们都是 Number 的子类。但还是不能往里面加元素,比如 list.add (10) 会报错 —— 因为编译器不知道具体是 Integer 还是 Double,怕加错类型。

再看 <? super T>,它代表 “T 及其父类”。比如 <? super Integer>,就包括 Number、Object 这些 Integer 的父类。这种写法的特点是 “能写不能读”(或者说读出来是 Object 类型)。比如你想写个方法,给 List 里加多个 Integer 元素:

public static void addIntegers (List<? super Integer> list) {

list.add (1);

list.add (2); // 没问题,因为?至少是 Integer 类型

}

调用的时候,传 List、List甚至 List都能行。但读元素的时候,只能用 Object 接收:

for (Object obj : list) {

// 要想用 Integer,得手动强转

Integer num = (Integer) obj;

}

小索奇之前做统计功能时,就用 <? super Integer> 存过数据 —— 既可以存到 Integer 列表,也能存到 Number 列表,灵活度特别高。

最后给你总结个小口诀,记起来更方便:“上界 extends 能读不能写,下界 super 能写不能读,通配符?只读不写,泛型读写都能搞”。

你们平时写代码时,有没有用过通配符却踩了坑的?比如想加元素加不进去,或者不知道该用 extends 还是 super?可以在评论区跟小索奇聊聊,咱们一起把泛型这点事儿彻底搞明白~

搜索即兴小索奇,点击关注,加入社区群聊,获取更多好用工具和资源

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

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

EXCEL开发之路(二)跨表交互模拟—仙盟创梦IDE

在车辆租赁行业&#xff0c;数据的高效管理与分析对于企业的运营决策、资源调配及客户服务优化至关重要。自建 Excel 实现多表统计交互&#xff0c;如同为行业装上了效能驱动引擎&#xff0c;助力企业在复杂多变的市场环境中稳健前行。一、精准资源管理&#xff0c;优化车辆调配…

医疗AI时代的生物医学Go编程:高性能计算与精准医疗的案例分析(八)

5.4 性能测试与结果分析 为了评估GoEHRStream的性能,我们设计测试模拟真实的医院数据流场景,并测量关键指标。 5.4.1 实验环境 硬件: CPU: Intel Xeon E-2288G (8 cores, 16 threads) RAM: 32 GB DDR4 Storage: 512 GB NVMe SSD (用于GoEHRStream和BadgerDB) Network: 1 G…

开关电源设计“反馈回路”部分器件分析

目录 主要分析问题如下&#xff1a; 一、问题1 二、问题二 分析电路如下&#xff1a; 主要分析问题如下&#xff1a; 1、分析TL431芯片1、2两引脚间并联电阻和电容&#xff08;RC电路&#xff09;的作用&#xff1f; 2、PC817A光耦输入两个引脚间并联电阻的作用&#xff1f;…

AI 编程新玩法:用 yunqi-saas-kit 框架制作小游戏,看广告变现轻松赚钱​

AI 编程新玩法&#xff1a;用 yunqi-saas-kit 框架制作小游戏&#xff0c;看广告变现轻松赚钱 在数字经济快速发展的当下&#xff0c;AI 技术正不断渗透到各个领域&#xff0c;其中 **#AI 编程凭借高效、便捷的优势&#xff0c;成为不少开发者和创业者的新选择。尤其是在小游戏…

Kafka 架构原理

一个kafka集群中包含一个或多个Producer、一个或多个broker、一个或多个ConsumerGrop以及一个Zookeeper集群。kafka通过Zookeeper管理kafka集群配置、leader副本的选举、生产者的负载均衡等。Producer使用push模式将消息发布到broker,Consumer使用pull模式从broker订阅并消费消…

用 PyTorch 搭建 CNN 实现 MNIST 手写数字识别

在图像识别领域&#xff0c;卷积神经网络&#xff08;CNN&#xff09; 凭借其对空间特征的高效提取能力&#xff0c;成为手写数字识别、人脸识别等任务的首选模型。而 MNIST&#xff08;手写数字数据集&#xff09;作为入门级数据集&#xff0c;几乎是每个深度学习学习者的 “第…

CTFshow系列——命令执行web61-68

本篇文章介绍了不同了方法进行题目的解析以及原因讲解。 文章目录Web61尝试了一下&#xff0c;被过滤的payload如下&#xff1a;所以&#xff0c;根据上述思路&#xff0c;这里尝试过的payload为&#xff1a;Web62&#xff08;同Web61&#xff09;Web63&#xff08;同Web62&…

.Net程序员就业现状以及学习路线图(二)

一、.NET程序员就业现状分析 1. 市场需求与岗位分布 2025年.NET开发岗位全国招聘职位约1676个&#xff0c;占全国技术岗位的0.009%&#xff0c;主要集中在一线城市如深圳、上海等地。就业单位类型分布为&#xff1a;软件公司占43.3%&#xff0c;研发机构占33.1%&#xff0c;物联…

MTK Linux DRM分析(二十二)- MTK mtk_drm_crtc.c(Part1)

一、代码分析 mtk_drm_crtc.c以mtk_crtc_comp_is_busy函数为界限进行拆分分析 static const struct drm_crtc_funcs mtk_crtc_funcs = {.set_config = drm_atomic_helper_set_config,.page_flip = drm_atomic_helper_page_flip,.destroy = mtk_drm_crtc_destroy,.reset = mtk…

stm32f103c8t6 led闪灯实验

目录 闪灯原理 2种接线方式控制闪灯 使用推挽接法 使用开漏接法 看原理图 写代码 闪灯原理 LED灯有个2-10mA的电流就可以点亮 3.3/5100.006A6mA 2种接线方式控制闪灯 使用推挽接法 当设置推挽模式时&#xff0c;CPU控制寄存器写0&#xff0c;IO引脚输出低电压&#xff0…

“我同意”按钮别乱点——你的“职业EULA”漏洞扫描报告

尊敬的审核&#xff1a; 本人文章《“我同意”按钮别乱点——你的“职业EULA”漏洞扫描报告》 1. 纯属技术交流&#xff0c;无任何违法内容 2. 所有法律引用均来自公开条文 3. 请依据《网络安全法》第12条“不得无故删除合法内容”处理 附&#xff1a;本文结构已通过区块链存证…

Product Hunt 每日热榜 | 2025-09-01

1. A01 标语&#xff1a;你个人的新闻助手 介绍&#xff1a;A01 是你的新闻助手&#xff0c;可以帮你关注你关心的任何话题。只需告诉它你想了解什么&#xff0c;它就能为你带来最新的文章。 产品网站&#xff1a; 立即访问 Product Hunt&#xff1a; View on Product Hunt…

【OpenFeign】基础使用

【OpenFeign】基础使用1. Feign介绍1.1 使用示例1.2 Feign与RPC对比1.3 SpringCloud Alibaba快速整合OpenFeign1.3.1 详细代码1. Feign介绍 1.什么是 Feign Feign 是 Netflix 开发的一个 声明式的 HTTP 客户端&#xff0c;在 Spring Cloud 中被广泛使用。它的目标是&#xff…

访问相同的url,相同入参的请求,Apifox/Postman可以正常响应结果,而本地调用不行(或结果不同)

文章目录问题概述Apifox查看实际请求总结问题概述 开发中有一个需求需要去别的系统中拿数据&#xff0c;配置好相关参数后发起请求时发现响应结果和在Apifox上不同&#xff0c;Apifox上正常显示数据&#xff0c;而本地调用后返回数据不存在。 这就很奇怪了&#xff0c;想了很多…

数据结构(C语言篇):(七)双向链表

目录 前言 一、概念与结构 二、双向链表的实现 2.1 头文件的准备 2.2 函数的实现 2.2.1 LTPushBack( )函数&#xff08;尾插&#xff09; &#xff08;1&#xff09;LTBuyNode( ) &#xff08;2&#xff09;LTInit( ) &#xff08;3&#xff09;LTPrint( ) &#x…

从拿起简历(resume)重新找工作开始聊起

经济萧条或经济衰退在经济相关学术上似乎有着严格的定义&#xff0c;我不知道我们的经济是否已经走向了衰退或者萧条&#xff0c;但有一点那是肯定的&#xff0c;那就现在我们的经济肯定是不景气的。经济不景气会怎么样&#xff1f;是的&#xff0c;会有很多人失业&#xff0c;…

OS+MySQL+(其他)八股小记

鲁迅先生曾经说过&#xff0c;每天进步一点点&#xff0c;妈妈夸我小天才。 依旧今日八股&#xff0c;这是我在多个文档整合一起的&#xff0c;可能格式有些问题&#xff0c;请谅解。 操作系统 1.进程和线程的区别&#xff1f; 进程是代码在数据集合的一次执行活动&#xff0c;…

Transformer的并行计算与长序列处理瓶颈总结

&#x1f31f; 第0层&#xff1a;极简版&#xff08;30秒理解&#xff09;一句话核心&#xff1a;Transformer像圆桌会议——所有人都能同时交流&#xff08;并行优势&#xff09;&#xff0c;但人越多会议越混乱&#xff08;长序列瓶颈&#xff09;。核心问题 并行优势&#x…

Vue 3 useId 完全指南:生成唯一标识符的最佳实践

&#x1f4d6; 概述 useId() 是 Vue 3 中的一个组合式 API 函数&#xff0c;用于生成唯一的标识符。它确保在服务端渲染&#xff08;SSR&#xff09;和客户端渲染之间生成一致的 ID&#xff0c;避免水合不匹配的问题。 &#x1f3af; 基本概念 什么是 useId&#xff1f; useId…

CGroup 资源控制组 + Docker 网络模式

1 CGroup 资源控制组1.1 为什么需要 CGroup - 容器本质 宿主机上一组进程 - 若无资源边界&#xff0c;一个暴走容器即可拖垮整机 - CGroup 提供**内核级硬限制**&#xff0c;比 ulimit、nice 更可靠1.2 核心概念 3 件套 | 概念 | 一句话解释 | 查看方式 | | Hierarchy | 树…