当前位置: 首页 > news >正文

137. Java 泛型 - 泛型与通配符:子类型化关系

文章目录

  • 137. Java 泛型 - 泛型与通配符:子类型化关系
    • **1. 常规类的继承与泛型类型的关系**
    • **2. 泛型类型与子类型化**
      • **❌ 错误示例:**
    • **3. 使用通配符解决泛型之间的子类型关系**
      • **✅ 示例:`List<Integer>` 和 `List<Number>` 之间的关系**
    • **4. 上限通配符与下限通配符的关系**
      • **上限通配符(`? extends T`)**
      • **下限通配符(`? super T`)**
    • **5. 使用通配符创建泛型类型之间的关系**
    • **6. 总结**
      • **通配符与泛型子类型化的要点:**
      • **通配符的实际应用:**
    • **7. 小贴士:通配符的使用场景**

137. Java 泛型 - 泛型与通配符:子类型化关系

在 Java 中,泛型类和接口并不像普通类那样直接遵循继承规则。为了解决这个问题,我们可以使用通配符 (? extends T? super T) 来创建泛型类或接口之间的关系。本篇将详细解释如何使用通配符建立泛型类型之间的子类型关系,并通过示例帮助您理解。


1. 常规类的继承与泛型类型的关系

在普通类中,继承遵循子类型规则,即如果类 B 扩展类 A,则 BA 的子类型。您可以这样编写代码:

class A { /* ... */ }
class B extends A { /* ... */ }B b = new B(); 
A a = b; // 合法,因为 B 是 A 的子类

🔍 子类型规则:

  • BA 的子类型,您可以将 B 类型的对象赋给 A 类型的变量。

2. 泛型类型与子类型化

与常规类不同,泛型类型之间并不直接遵循子类型化规则。这意味着,即使 IntegerNumber 的子类,List<Integer> 也不是 List<Number> 的子类型,实际上这两者之间没有任何继承关系。

❌ 错误示例:

List<B> lb = new ArrayList<>();
List<A> la = lb;   // 编译错误:List<B] 不是 List<A> 的子类型

即使 BA 的子类,List<B> 并不等于 List<A>,这会导致编译错误。


3. 使用通配符解决泛型之间的子类型关系

为了建立泛型类型之间的关系,可以使用上限通配符 (? extends T) 来让 Java 编译器理解某个泛型类型是另一个泛型类型的子类型。

✅ 示例:List<Integer>List<Number> 之间的关系

IntegerNumber 的子类型,但是 List<Integer> 不是 List<Number> 的子类型。这是因为 泛型类型的子类型化并不自动传递。我们可以通过上限通配符来为 List<Integer>List<Number> 之间建立关系。

List<? extends Integer> intList = new ArrayList<>();
// 这是合法的,因为 List<? extends Integer> 是 List<? extends Number> 的子类型
List<? extends Number> numList = intList;

🔍 解析:

  • intListList<? extends Integer>,它表示存储了 Integer 或其子类的列表。
  • numListList<? extends Number>,它表示存储了 Number 或其子类的列表。
  • 由于 IntegerNumber 的子类,所以 List<? extends Integer> 也可以看作 List<? extends Number> 的子类型。这样,我们就可以建立起 intListnumList 之间的关系。

4. 上限通配符与下限通配符的关系

上限通配符(? extends T

  • 用途:用来指定类型范围,表示元素的类型是 TT 的子类型。
  • 适用场景:适用于读取元素(数据消费)场景,您只能读取 T 类型的数据,不能插入(除了 null)。

下限通配符(? super T

  • 用途:用来指定类型范围,表示元素的类型是 TT 的超类型。
  • 适用场景:适用于写入元素(数据生产)场景,您可以将 T 类型的数据插入列表中,但只能作为 Object 类型读取元素。

5. 使用通配符创建泛型类型之间的关系

通过通配符,我们可以创建泛型类型之间的关系,使得代码更加灵活。以下是一些关键的泛型类型关系:

  • List<? extends Number> 可以由 List<Integer>List<Double> 等列表来填充,因为它们都是 Number 的子类。
  • List<? super Integer> 可以由 List<Integer>List<Number>List<Object> 来填充,因为 Integer 是这些类型的父类。
List<? extends Number> numList = new ArrayList<Integer>(); // 合法,因为 Integer 是 Number 的子类
List<? super Integer> objList = new ArrayList<Number>();  // 合法,因为 Integer 是 Number 的子类

6. 总结

通配符与泛型子类型化的要点:

  • 普通类的继承:遵循子类型规则,如果 B 扩展 A,则 BA 的子类型。
  • 泛型类的继承:泛型类型之间没有直接的继承关系,List<B> 并不是 List<A> 的子类型。
  • 使用通配符(? extends T? super T
    • ? extends T:表示“某个类型 TT 的子类型”,适用于数据读取(消费数据)。
    • ? super T:表示“某个类型 TT 的超类型”,适用于数据写入(生产数据)。

通配符的实际应用:

  • 建立泛型类型之间的关系,让代码更加灵活。
  • 使用通配符来处理不同类型的泛型数据,实现更广泛的类型兼容性

7. 小贴士:通配符的使用场景

  • ? extends T:适用于 读取数据 时需要限制元素类型范围的情况,数据消费者
  • ? super T:适用于 写入数据 时需要限制元素类型范围的情况,数据生产者

🎯 记住

  • 泛型类型之间并没有直接的继承关系,但通配符可以帮助建立这些关系,使得代码更灵活。
http://www.dtcms.com/a/288885.html

相关文章:

  • 【Linux】权限详解 权限本质、权限属性、su、sudo提权、chmod\chown\chgrp、文件类别
  • RxSwift-事件属性
  • Vuex 核心知识详解:Vue2Vue3 状态管理指南
  • Servlet快速入门
  • Docker在NAS部署MoonTV+OrionTV
  • 44.sentinel授权规则
  • tidyverse-数据读入
  • 基于智慧经营系统的学校住宿登记报表分析与应用探究-毕业论文—仙盟创梦IDE
  • 如何防止任务长期处于“等待”状态
  • Python基础和高级【抽取复习】
  • 基于单片机的自动条幅悬挂机
  • Leetcode 06 java
  • SpringBoot的配置文件
  • 【micro:bit】从入门到放弃(四):高级指令:函数、数组、文本、高级LED、引脚配置
  • UE蒙太奇和动画序列有什么区别?
  • 早期SD模型控制生成方法
  • Dev-C++——winAPI贪吃蛇小游戏
  • 智能制造之物料详解
  • 物联网安装调试-温湿度传感器
  • 数学专业转行做大数据容易吗?需要补什么?
  • 高性能熔断限流实现:Spring Cloud Gateway 在电商系统的实战优化
  • 本地部署 Claude 大语言模型的完整实践指南
  • 从磁记录到数据中心:磁盘原理与服务器架构的完整技术链路
  • 【C++基础】面试高频考点解析:extern “C“ 的链接陷阱与真题实战
  • 【原创】微信小程序添加TDesign组件
  • 网络安全初级(前端页面的编写分析)
  • 手写tomcat
  • JAVA学习-练习试用Java实现“贝叶斯网络:实现一个简单的贝叶斯网络用于因果推理”
  • Transient Storage
  • Redis通用常见命令(含面试题)