Java基础 - 泛型(常见用法)

文章目录

  • 泛型类
  • 泛型方法
  • 泛型类派生子类
    • 示例 1:子类固定父类泛型类型(`StringBox` 继承自 `Box<String>`)
    • 示例 2:子类保留父类泛型类型(`AdvancedBox<T>` 继承自 `Box<T>`)
    • 示例 3:添加子类自己的泛型参数(`KeyValuePair<K,V>` 继承自 `Pair<K>`)
    • 示例 4:约束父类类型边界(`IntCalculator` 继承自 `NumberCalculator<Integer>`)
  • 类型通配符
    • 示例 1:无界通配符(`<?>`)
    • 示例 2:上界通配符(`<? extends Number>`)
    • 示例 3:下界通配符(`<? super Integer>`)

泛型类

泛 型 类 就 是 把 泛 型 定 义 在 类 上 , 用 户 使 用 该 类 的 时 候 , 才 把 类 型 明 确 下 来 。 这样的话, 用户明确了什么类型 , 该类就代表着什么类型 , 用户在使用的时候就不用担心强转的问题, 和运行时转换异常的问题了。

public class Box<T> {private T content;  // 泛型字段public void setContent(T content) {this.content = content;}public T getContent() {return content;}public static void main(String[] args) {// 存储字符串类型Box<String> stringBox = new Box<>();stringBox.setContent("Hello Generics");System.out.println(stringBox.getContent()); // 输出: Hello Generics// 存储整数类型Box<Integer> intBox = new Box<>();intBox.setContent(100);int value = intBox.getContent(); // 无需强制类型转换System.out.println(value);       // 输出: 100}
}

关键点:

  • Box<T> 的 T 是类型参数,使用时由开发者指定具体类型(如 String 或 Integer)。
  • 类型安全:例如 stringBox.setContent(100) 会在编译时报错。

泛型方法

除了在类上使用泛型 ,我们可能就 仅 仅 在 某 个 方 法 上 需 要 使 用 泛 型 , 外界仅仅是关心该方法 , 不关心类其他的属性 , 这样的话 , 我们在整个类上定义泛型 , 未免就有些大题小作了 。 那么此时 , 我们可以采用泛型方法。

public class MaxFinder {/*** 泛型方法:找到数组中的最大值(要求元素类型实现 Comparable 接口)* @param array 输入数组* @return 最大值*/public static <T extends Comparable<T>> T findMax(T[] array) {if (array == null || array.length == 0) {throw new IllegalArgumentException("数组不能为空");}T max = array[0];for (T item : array) {if (item.compareTo(max) > 0) {max = item;}}return max;}public static void main(String[] args) {Integer[] integers = {5, 8, 2, 10};Integer maxInt = findMax(integers); // 调用时类型为 IntegerSystem.out.println("Max Integer: " + maxInt); // 输出: 10String[] fruits = {"Apple", "Banana", "Mango"};String maxFruit = findMax(fruits); // 调用时类型为 StringSystem.out.println("Max String: " + maxFruit); // 输出: Mango(按字典序比较)}
}

关键点:

  • <T> 声明在方法返回类型前,表示这是一个泛型方法,独立于类是否泛型。
  • <T extends Comparable<T>> 确保类型 T 的对象可以相互比较(通过 compareTo 方法)。
  • 类型安全:拒绝不可比较的类型:例如 findMax(new Object[]{…}) 会编译失败。
  • 返回 T 类型,确保返回元素类型与输入数组类型完全一致。

泛型类派生子类

前 面 我 们 已 经 定 义 了 泛型 类 , 泛 型 类 是 拥有泛型特性的类 , 它本质上还 是 一 个 J av a 类 , 那 么 它 就 可 以 被 继 承 或 实 现 。这个情况比较多:

示例 1:子类固定父类泛型类型(StringBox 继承自 Box<String>

子类直接指定父类泛型参数的具体类型,适用于特定场景的扩展:

// 泛型父类
class Box<T> {private T content;public void setContent(T content) {this.content = content;}public T getContent() {return content;}
}
// 子类继承时指定父类泛型类型为 String
class StringBox extends Box<String> {// 新增方法:专为字符串内容设计public void toUpperCase() {String value = getContent(); // 直接使用 String 类型if (value != null) { setContent(value.toUpperCase());}}
}
public class Main {public static void main(String[] args) {StringBox box = new StringBox();box.setContent("hello");box.toUpperCase();System.out.println(box.getContent()); // 输出: HELLO}
}

关键点:

  • 子类 StringBox 继承时固定了父类泛型为 String,因此 getContent() 直接返回 String 类型。
  • 子类可以添加与固定类型相关的专属方法(如 toUpperCase)。

示例 2:子类保留父类泛型类型(AdvancedBox<T> 继承自 Box<T>)

子类保持父类泛型参数的灵活性,并扩展新功能:

// 父类定义不变,与示例1相同// 子类保留父类的泛型参数 <T>
class AdvancedBox<T> extends Box<T> {// 新增方法:检查内容是否为空public boolean isEmpty() {return getContent() == null;}// 新增方法:重置内容为 nullpublic void clear() {setContent(null);}
}public class Main2 {public static void main(String[] args) {AdvancedBox<Integer> intBox = new AdvancedBox<>();intBox.setContent(100);System.out.println(intBox.isEmpty()); // 输出: falseintBox.clear();System.out.println(intBox.getContent()); // 输出: null}
}

关键点:

  • 子类 AdvancedBox<T> 保留父类泛型参数 <T>,可复用父类逻辑。
  • 功能上扩展了新方法(如 isEmpty 和 clear),独立于具体类型的通用操作。

示例 3:添加子类自己的泛型参数(KeyValuePair<K,V> 继承自 Pair<K>

父子类均使用泛型,子类引入新的类型参数:

// 泛型父类
class Pair<T> {private T first;private T second;public Pair(T first, T second) {this.first = first;this.second = second;}public T getFirst() { return first; }public T getSecond() { return second; }
}// 子类添加新的泛型参数 V,与父类参数 K 独立
class KeyValuePair<K, V> extends Pair<K> {private V value;public KeyValuePair(K key, V value) {super(key, key); // 父类需要两个同类型参数,此处复用 key 作为示例this.value = value;}// 新增方法:获取键值对的独立值public V getValue() { return value; }
}public class Main3 {public static void main(String[] args) {KeyValuePair<String, Integer> entry = new KeyValuePair<>("Age", 30);String key = entry.getFirst(); // 类型为 String(父类返回值)int value = entry.getValue();  // 类型为 Integer(子类新增方法)System.out.println(key + ": " + value); // 输出: Age: 30}
}

关键点:

  • 子类 KeyValuePair<K, V> 扩展了父类的泛型参数 <K>,并新增了独立参数 <V>
  • 父子类泛型参数独立,子类可以实现更复杂的数据结构(如键值对)。

示例 4:约束父类类型边界(IntCalculator 继承自 NumberCalculator<Integer>

子类继承时遵循父类的泛型约束(如 <T extends Number>),并进一步具体化:

// 泛型父类,约束类型必须为 Number 的子类
class NumberCalculator<T extends Number> {protected T value;public NumberCalculator(T value) {this.value = value;}public double doubleValue() {return value.doubleValue();}
}// 子类指定父类泛型类型为 Integer
class IntCalculator extends NumberCalculator<Integer> {public IntCalculator(Integer value) {super(value);}// 新增方法:计算平方(专为 Integer 设计)public int squareInt() {return value * value;}
}public class Main4 {public static void main(String[] args) {IntCalculator calc = new IntCalculator(5);System.out.println(calc.doubleValue()); // 输出: 5.0(调用父类方法)System.out.println(calc.squareInt());   // 输出: 25(子类专属方法)}
}

关键点:

  • 父类 NumberCalculator<T> 的泛型参数已被约束为 <T extends Number>
  • 子类 IntCalculator 继承时将泛型固定为 Integer,确保父类逻辑安全,同时添加 Integer 专用的新方法。

类型通配符

示例 1:无界通配符(<?>

用于处理未知类型的集合,适合只读取集合内容的场景:

public class UnboundedWildcardDemo {/*** 打印任意类型集合的元素* @param list 使用无界通配符 <?>,表示接受任意类型的 List*/public static void printList(List<?> list) {for (Object item : list) {System.out.print(item + " ");}System.out.println();}public static void main(String[] args) {List<String> names = Arrays.asList("Alice", "Bob", "Charlie");List<Integer> numbers = Arrays.asList(1, 2, 3);printList(names);   // 输出: Alice Bob Charlie printList(numbers); // 输出: 1 2 3 }
}

关键点:

  • List<?> 可以接受任何类型的 List,如 List<String>List<Integer>
  • 只能读取元素(返回 Object),但 不能添加 元素(除了 null),因为具体类型未知。

示例 2:上界通配符(<? extends Number>

限制类型为 Number 或其子类(如 Integer、Double),适合读取场景:

public class UpperBoundedWildcardDemo {/*** 计算数值列表的总和* @param list 使用上界通配符 <? extends Number>* @return Sum of numbers as double*/public static double sumOfList(List<? extends Number> list) {double sum = 0.0;for (Number num : list) {sum += num.doubleValue(); // 调用 Number 的方法}return sum;}public static void main(String[] args) {List<Integer> integers = Arrays.asList(1, 2, 3);List<Double> doubles = Arrays.asList(1.1, 2.2, 3.3);System.out.println(sumOfList(integers)); // 输出: 6.0System.out.println(sumOfList(doubles));  // 输出: 6.6}
}

关键点:

  • 参数 list 可以接受 List<Integer>List<Double> 等(只要元素是 Number 的子类)。
  • 不能添加元素(如 list.add(5) 会编译报错),具体子类型未知,可能破坏类型安全。

示例 3:下界通配符(<? super Integer>

限制类型为 Integer 或其父类(如 Number、Object),适合写入场景:

public class LowerBoundedWildcardDemo {/*** 向集合中添加多个 Integer 值* @param dest 使用下界通配符 <? super Integer>,可以接受Integer或其父类(如Number、Object)的List* @param values 要添加的多个 Integer 值*/public static void addNumbersToList(List<? super Integer> dest, List<Integer> values) {dest.addAll(values); }public static void main(String[] args) {List<Number> numberList = new ArrayList<>();addNumbersToList(numberList, Arrays.asList(10, 20, 30));System.out.println(numberList); // 输出: [10, 20, 30]List<Object> objectList = new ArrayList<>();addNumbersToList(objectList, Arrays.asList(100, 200));System.out.println(objectList); // 输出: [100, 200]List<? super Integer> list = new ArrayList<Number>();list.add(1); // 允许Object obj = list.get(0); // 只能作为Object类型读取}
}

关键点:

  • 参数 dest 可以是 List、List 等(父类型容器)。
  • 允许添加 Integer 或其子类型的元素。

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

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

相关文章

YOLO学习笔记 | YOLOv8环境搭建全流程指南(2025.4)

===================================================== github:https://github.com/MichaelBeechan CSDN:https://blog.csdn.net/u011344545 ===================================================== YOLOv8环境搭建 一、环境准备与工具配置1. Conda虚拟环境搭建2. CUDA与…

【 Beautiful Soup (bs4) 详解】

引言 Beautiful Soup 是 Python 最流行的 HTML/XML 解析库&#xff0c;能够从复杂的网页文档中高效提取数据。以下是其核心知识点及示例代码。 一、库简介 1. 核心模块 BeautifulSoup&#xff1a;主类&#xff0c;用于构建文档树结构Tag&#xff1a;表示 HTML/XML 标签的对象…

傅利叶发布首款开源人形机器人N1:开发者可实现完整复刻

2025年4月11日&#xff0c;上海——通用机器人公司傅利叶正式发布首款开源人形机器人 Fourier N1&#xff0c;并同步开放涵盖物料清单、设计图纸、装配指南、基础操作软件在内的完整本体资源包。作为傅利叶 “Nexus 开源生态矩阵” 的首个落地项目&#xff08;“N1” 即 “Nexu…

视觉目标检测大模型GAIA

中国科学院自动化研究所智能感知与计算研究中心携手华为等领军企业&#xff0c;共同推出面向产业应用的视觉目标检测全流程解决方案——GAIA智能检测平台。该研究成果已获CVPR 2021会议收录&#xff08;论文链接&#xff1a; 论文地址&#xff1a;https://arxiv.org/pdf/2106.…

前端时间同步利器:React + useEffect 实现高性能动态时钟

前言 在你奋笔疾敲代码的瞬间&#xff0c;是不是突然一低头&#xff0c;发现时间像偷偷跑路的变量&#xff0c;一眨眼就从上午飘到下午&#xff1f;饭没吃、会没开、工位也快被前端猫霸占了。仿佛你写的不是代码&#xff0c;而是“时间穿梭机”。别慌&#xff0c;咱们今天就来…

前端动画性能优化

前端动画性能优化全攻略&#xff1a;告别卡顿与高CPU占用 一、动画性能问题现状分析 1.1 性能问题现象 动画帧率低于60FPS时出现明显卡顿滚动/缩放操作时响应延迟CPU占用率长期超过70%移动端设备发热严重 1.2 核心问题根源 浏览器渲染流程中的性能瓶颈主要出现在&#xff1…

springboot中如何处理跨域

什么是跨域 跨域&#xff08;Cross-Origin&#xff09;是浏览器出于安全考虑&#xff0c;对不同源的资源访问施加的限制机制。其核心原因是同源策略&#xff08;Same-Origin Policy&#xff09;&#xff0c;即浏览器仅允许协议&#xff08;Protocol&#xff09;、域名&#xf…

js实现生肖宜忌展示

实现效果图如下 实现逻辑&#xff1a; 1.录入属相列表&#xff08;列表顺序不可调整&#xff09;&#xff1b; 2.录入各属相相宜、相忌属相&#xff1b; 3.输入年份后&#xff0c;根据属相列表获取到正确的属相&#xff1b; 4.根据获取的属相去展示宜、忌属相&#xff1b; 5.打…

3DMAX笔记-UV知识点和烘焙步骤

1. 在展UV时&#xff0c;如何点击模型&#xff0c;就能选中所有这个模型的uv 2. 分多张UV时&#xff0c;不同的UV的可以设置为不同的颜色&#xff0c;然后可以通过颜色进行筛选。 3. 烘焙步骤 摆放完UV后&#xff0c;要另存为一份文件&#xff0c;留作备份 将模型部件全部分成…

AI 重构 Java 遗留系统:从静态方法到 Spring Bean 注入的自动化升级

在当今快速发展的软件行业中&#xff0c;许多企业都面临着 Java 遗留系统的维护和升级难题。这些老旧系统往往采用了大量静态方法&#xff0c;随着业务的不断发展&#xff0c;其局限性日益凸显。而飞算 JavaAI 作为一款强大的 AI 工具&#xff0c;为 Java 遗留系统的重构提供了…

【从一个 TypeScript 报错理解 ES6 模块的三种导入方式】

从一个 TypeScript 报错理解 ES6 模块的三种导入方式 在日常开发中&#xff0c;我们经常遇到模块导入导出的场景。最近在处理一个项目时&#xff0c;遇到了一个有趣的问题&#xff1a;对于只有默认导出的模块&#xff0c;我们该使用哪种导入方式&#xff1f;这个问题引发了对 …

安徽京准:NTP网络时钟服务器功能及同步模式的介绍

安徽京准&#xff1a;NTP网络时钟服务器功能及同步模式的介绍 安徽京准&#xff1a;NTP网络时钟服务器功能及同步模式的介绍 1、NTP网络时钟服务器概念&#xff1a; NTP时钟服务器&#xff0c;表面意思是时间计量工具的服务设备&#xff0c;其在现代工业中是用于对客户端设备…

JMeter从入门到荒废-常见问题汇总

启动某个ThreadGroup的时候&#xff0c;启动不了 现象 点击start按钮的时候&#xff0c;结果树和汇总报告都没有任何数据。 同时&#xff0c;点击右上角的error log 发现有错误信息&#xff1a; 错误信息如下&#xff1a; 2025-04-09 10:03:48,009 ERROR o.a.j.g.a.ActionR…

Elasticsearch 学习规划

Elasticsearch 学习规划 明确学习目标与动机 场景化需求分析 - **S**&#xff1a;掌握Elasticsearch架构体系&#xff0c;熟练使用Elasticsearch 进行数据分析,Elasticsearch结合java 项目落地案例 - **M**&#xff1a;搜索和Elasticsearch相关GitHub项目 - **A**&#xff1a;每…

核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室

核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室 为满足当今无人机行业应用需求&#xff0c;推动无人机技术的教育与实践深度融合&#xff0c;北京卓翼智能科技有限公司旗下品牌飞思实验室与湖南汽车工程职业大学强强联手&#xff0c;共同建设无人机操控与编队技术…

【Android】Android 获取当前前台应用包名与自动化控制全流程实践笔记(适配 Android 10+)

一、前言 在 Android 系统中&#xff0c;获取当前运行的前台应用、返回桌面、跳转权限设置、关闭其他应用等行为&#xff0c;往往受到系统的严格限制。随着 Android 版本的提升&#xff08;特别是 Android 10 之后&#xff0c;即 API 29&#xff09;&#xff0c;很多传统方法已…

Sentinel核心源码分析(上)

文章目录 前言一、客户端与Spring Boot整合二、SphU.entry2.1、构建责任链2.2、调用责任链2.2.1、NodeSelectorSlot2.2.2、ClusterBuilderSlot2.2.3、LogSlot2.2.4、StatisticSlot2.2.5、AuthoritySlot2.2.6、SystemSlot2.2.7、FlowSlot2.2.7.1、selectNodeByRequesterAndStrat…

浅谈「分词」:原理 + 方案对比 + 最佳实践

在文本搜索、自然语言处理、智能推荐等场景中&#xff0c;「分词」 是一个基础但至关重要的技术点。无论是用数据库做模糊查询&#xff0c;还是构建搜索引擎&#xff0c;分词都是提高效率和准确度的核心手段。 &#x1f50d; 一、什么是分词&#xff1f; 分词&#xff08;Tok…

transformers:打造的先进的自然语言处理

github地址&#xff1a;https://github.com/huggingface/transformers Transformers 提供了数以千计的预训练模型&#xff0c;支持 100 多种语言的文本分类、信息抽取、问答、摘要、翻译、文本生成。它的宗旨是让NLP 技术人易用。 Transformers 提供了便于快速下载和使用的API…

Spring Boot 集成 MongoDB 时自动创建的核心 Bean 的详细说明及表格总结

以下是 Spring Boot 集成 MongoDB 时自动创建的核心 Bean 的详细说明及表格总结&#xff1a; 核心 Bean 列表及详细说明 1. MongoClient 类型&#xff1a;com.mongodb.client.MongoClient作用&#xff1a; MongoDB 客户端核心接口&#xff0c;负责与 MongoDB 服务器建立连接、…