Java泛型核心知识总结

目录

  • Java 泛型核心知识总结
    • 泛型
      • 什么是泛型?有什么用?
      • 泛型有哪些限制?为什么?
      • 项目中哪里用到了泛型?
      • 什么是类型擦除?
      • 什么是桥方法?
    • 通配符
      • 什么是通配符?有什么作用?
      • 通配符 ? 和常用的泛型 T 之间有什么区别?
      • 介绍一下常用的通配符?
    • 学习参考

Java 泛型核心知识总结

泛型

什么是泛型?有什么用?

Java 泛型(Generics)是 JDK 5 中引入的一个新特性,它提供了一种类型安全的编程机制,可以在编译时检查类型错误,避免了在运行时出现类型转换异常的情况。它可以使程序员在编写代码时指定类型参数,从而使得代码更加灵活和可重用。

比如 ArrayList<Persion> persons = new ArrayListPersion>() 这行代码就指明了该 ArrayList 对象只能传入 Persion 对象,如果传入其他类型的对象就会报错。

类型参数 T 是一种占位符类型,用于表示实际的类型。

一般在哪定义?

Java 泛型可以用于类、接口和方法的定义中。

三种使用方式:泛型类、泛型接口、泛型方法。

使用泛型有什么好处?

简单来说,使用泛型参数,可以增强代码的可读性以及稳定性。

使用泛型可以带来许多好处,比如:

  • 类型安全:Java 泛型可以在编译时检查类型错误,避免了在运行时出现类型转换异常的情况。
  • 代码重用:泛型可以使代码更加通用和模块化,可以重用在不同的场景中。
  • 更好的性能:泛型可以避免不必要的类型转换和装箱操作,从而提高代码的性能。

泛型的实现方式?

泛型主要通过以下两种方式来实现:参数化类型和通配符类型。

泛型有哪些限制?为什么?

  1. 泛型参数 <T> 不能是基本类型,例如 int,因为实际类型是 ObjectObject 类型无法持有基本类型。
  2. 无法取得带泛型的 Class。(getClass()
  3. 无法判断带泛型的类型。因为擦除后为 object 类型,object 无法使用 > 进行比较。
  4. 不能实例化泛型参数的数组。因为擦除后为 object 后无法进行类型判断。
  5. 不能实现两个不同泛型参数的同一接口,擦除后多个父类的桥方法将冲突。
  6. 不能使用 static 修饰泛型变量。
  7. 泛型数组的限制:Java 泛型数组的创建和使用受到一些限制,例如无法创建泛型数组、无法向泛型数组中添加元素等。

项目中哪里用到了泛型?

比如:

  • 自定义接口通用返回结果类 CommonResult<T> 通过参数 T 可根据具体的返回类型动态指定结果的数据类型
  • 定义 Excel 处理类 ExcelUtil<T> 用于动态指定 Excel 导出的数据类型
  • 构建集合工具类(参考 Collections 中的 sort, binarySearch 方法)。

什么是类型擦除?

类型擦除是指在 Java 编译器将泛型代码编译成字节码时,会将泛型类型擦除,替换为实际的类型或者 Object 类型,从而使得泛型类型在运行时不存在

注意:是【实际的类型】或者 【Object】类型。

为什么要擦除?

这是因为 Java 虚拟机并不支持泛型,所以需要在编译期对泛型进行擦除,将泛型代码转换为普通的 Java 类型。

比如:

泛型类型没有定义类型参数的限定类型的情况:

class MyList<T> { ... }

对于上面的 MyList<T> 类,实际上编译器会将其擦除成如下形式

class MyList {...
}

在运行时,我们无法获取泛型类型的类型参数,例如 无法获取 MyList<String>MyList<Integer> 的类型参数,它们都被擦除为 MyList 类型。

什么是桥方法?

桥方法(Bridge Method)是 Java 泛型类型擦除机制的一种补偿措施。它是指在泛型类或泛型接口中,由编译器自动生成的一个方法,用于在类型擦除后保持多态性

具体说明:

在 Java 泛型中,由于类型擦除机制的存在,导致在某些情况下,泛型类型的继承关系会被破坏。

例如下面的代码:

public class MyList<T> {...public void add(T element) {...}
}

假设我们定义了一个子类:

public class MyStringList extends MyList<String> {...
}

由于类型擦除机制的存在,MyStringList 类实际上是继承自 MyList 类的原始类型,而不是继承自 MyList<String> 类型。

因此,如果我们在 MyStringList 类中定义一个重写 add() 方法的话,会出现编译错误:

public class MyStringList extends MyList<String> {...@Overridepublic void add(String element) {...}
}

这是因为 Java 编译器会将 MyStringList 类中的 add() 方法擦除成如下形式

public void add(Object element) {...
}

这个方法的参数类型是 Object,与 MyList<String> 中的 add 方法的参数类型不同,因此编译器会报错。

为了解决这个问题,Java 编译器会在 MyStringList 类中自动生成一个桥方法,用于在类型擦除后保持多态性:

public class MyStringList extends MyList<String> {...@Overridepublic void add(Object element) {add((String) element);}public void add(String element) {...}
}

在上面的代码中,编译器会自动生成一个桥方法 add(Object element)它会调用原始方法 add(String element),从而保持多态性

通配符

什么是通配符?有什么作用?

泛型类型是固定的,某些场景下使用起来不太灵活,于是,通配符就来了!

通配符可以允许类型参数变化,用来解决泛型无法协变的问题。

举个例子:

// 限制类型为 Person 的子类
<? extends Person>
// 限制类型为 Manager 的父类3
<? super Manager>

通配符 ? 和常用的泛型 T 之间有什么区别?

  1. T 可以用于声明变量或常量,而 ? 不行。
  2. T 一般用于声明泛型类或方法,通配符 ? 一般用于泛型方法的调用代码和形参。
  3. T编译期会被擦除为限定(实际)类型或 object,通配符用于捕获具体类型。

擦除为限定(实际)类型是什么意思?

1、如果泛型类型定义了类型参数的限定类型,例如:

class MyList<T extends Number> {private T[] elements;public void add(T element) { ... }public T get(int index) { ... }
}

那么在编译时,编译器会将类型参数 T 擦除为其限定类型 Number,例如:

class MyList {private Number[] elements;public void add(Number element) { ... }public Number get(int index) { ... }
}

因此,在运行时,无法获取泛型类型的类型参数 T,而只能获取其限定类型 Number

2、如果泛型类型没有定义类型参数的限定类型,例如:

class MyList<T> {private T[] elements;public void add(T element) { ... }public T get(int index) { ... }
}

那么在编译时,编译器会将类型参数 T 擦除为 Object 类型,例如:

class MyList {private Object[] elements;public void add(Object element) { ... }public Object get(int index) { ... }
}

因此,在运行时,无法获取泛型类型的类型参数 T,而只能获取其擦除后的类型 Object

介绍一下常用的通配符?

常用的通配符有三种:

  1. <? extends T>上边界通配符 extends,表示该泛型必须是 T 的子类(包括 T 本身),用于限定泛型的上界。
  2. <? super Integer>下边界通配符 super,表示该泛型必须是 T 的父类(包括 T 本身),用于限定泛型的下界。
  3. <?>无限定通配符 ?,表示任意类型,用于表示不确定的类型参数。无限定通配符 <?> 很少使用,可以用 <T> 替换,同时它是所有 <T> 类型的父类。

学习参考

一文搞懂泛型:https://www.cnblogs.com/XiiX/p/14719568.html

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

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

相关文章

四、Lua循环

文章目录 一、while(循环条件)二、for&#xff08;一&#xff09;数值for&#xff08;二&#xff09;泛型for&#xff08;三&#xff09;repeat util 既然同为编程语言&#xff0c;那么控制逻辑里的循环就不能缺少&#xff0c;它可以帮助我们实现有规律的重复操作&#xff0c;而…

AIGC-文生视频

stable diffusion&#xff1a; stable diffusion原理解读通俗易懂&#xff0c;史诗级万字爆肝长文&#xff0c;喂到你嘴里 - 知乎个人网站一、前言&#xff08;可跳过&#xff09;hello&#xff0c;大家好我是 Tian-Feng&#xff0c;今天介绍一些stable diffusion的原理&#…

【Python基础】字符集与字符编码

先行了解的知识&#xff1a; 1. 编码和解码 计算机内存储的信息都是二进制表示。 我们看到的英文&#xff0c;数字&#xff0c;汉字等在计算机内如何表示&#xff0c;那就需要编码 计算机内存储的信息需要解析出来&#xff0c;那就是解码 2.字符集与分类 什么是字符集&#xf…

力扣5.最长回文子串

题目描述 思路 1.能够反复利用已判断好的回文子串 2.当子串s[i1,j-1]是回文子串时&#xff0c;只要s[i]s[j]&#xff0c;那么s[i,j]也会是回文子串 3.用好动态规划&#xff0c;具体解释在代码注释里 代码 class Solution {public String longestPalindrome(String s) {int…

Redis分布式锁学习总结

⭐️ 前言 想必大家都有过并发编程的经验&#xff0c;在一个单体应用中&#xff0c;可以通过java提供的各种锁机制来控制多线程对于单体应用中同一资源的并发访问&#xff1b;那么在分布式场景下&#xff0c;想要控制多个应用对于同一外部资源的并发访问&#xff0c;就要用到分…

利用python实现文件压缩打包的功能

主要是利用了zipfile实现文件压缩打包&#xff0c;简单实例代码如下&#xff1a; import zipfilewith zipfile.ZipFile("archive.zip",w) as zipf:zipf.write("config.ini")zipf.write("test.py") 其中的模式 w表示如果没有该文件则创建该文件…

HarmonyOS4.0开发应用(二)【快速学习】

快速学习 创建项目 1.开始创建 2.选择模板 刚开始选择空白的模板即可 3.填写项目信息 这样一个基本项目就创建好了 代码结构 实现Demo(文字动态切换) Entry Component struct Index {State message: string Hello Worldbuild() {Row() {Column() {Text(this.message).fo…

学习笔记三十五:Ingress-controller高可用

Ingress-controller高可用 Ingress-controller高可用特别注意&#xff1a; 通过keepalivednginx实现ingress-nginx-controller高可用安装nginx主备&#xff1a;修改nginx配置文件。主备一样keepalive配置主keepalived备keepalivek8snode1和k8snode2启动服务测试vip是否绑定成功…

网站实现验证码功能

一、验证码 一般来说&#xff0c;网站在登录的时候会生成一个验证码来验证是否是人类还是爬虫&#xff0c;还有一个好处是防止恶意人士对密码进行爆破。 二、流程图 三、详细说明 3.1 后端生成验证码 Override public Result<Map<String, String>> getVerifica…

语音信号处理:librosa

1 librosa介绍 Librosa是一个用于音频和音乐分析的Python库&#xff0c;专为音乐信息检索&#xff08;Music Information Retrieval&#xff0c;MIR&#xff09;社区设计。自从2015年首次发布以来&#xff0c;Librosa已成为音频分析和处理领域中最受欢迎的工具之一。它提供了一…

Vlan配置

需求 1 PC1和PC3所在接口为Access接口 PC2/4/5/6处于同一网段&#xff0c;其中pc2可以访问pc4/5/6 PC4可以访问pc5&#xff0c;但不能访问pc6 PC5不能访问PC6 2 PC1/3与PC2/4/5/6不再同一网段 3 所有PC通过DHCP获取IP地址&#xff0c;且PC1/3可以正常访问PC2/4/5/6 R1 [V200R00…

关于页面文件

project.config.json文件 Setting&#xff1a;保存了编译相关的配置 projectname&#xff1a;保存项目名称 appid&#xff1a;保存了小程序的账号 Sitemap.json文件 用来配置小程序页面是否允许微信索引 "rules":[{"action":"allow", //a…

python使用记录

1、VSCode添加多个python解释器 只需要将对应的python.exe的目录&#xff0c;添加到系统环境变量中即可&#xff0c;VSCode会自动识别及添加 2、pip 使用 pip常用命令和一些坑 查看已安装库的版本号 pip show 库名称 通过git 仓库安装第三方库 pip install git仓库地址

AI时代架构师之路:技术、洞察和创新的完美融合

随着人工智能技术的飞速发展&#xff0c;我们正置身于一个由数据驱动的时代。在这个充满无限可能性的AI时代&#xff0c;架构师成为设计和构建先进系统的关键角色。然而&#xff0c;在追逐技术的同时&#xff0c;架构师需要修炼一系列综合素养&#xff0c;使其在技术、业务和伦…

软文写作的核心技巧,媒介盒子分享

软文作为广告领域中的一种重要方式&#xff0c;其创意和写作技巧对于品牌推广来说至关重要&#xff0c;但是软文并不是简单几句就能拿捏用户&#xff0c;还需要掌握其中的内在逻辑&#xff0c;今天媒介盒子就来和大家聊聊&#xff1a;如何写好软文。 一、 文案创作三要素 虽然…

对某登录站点的JS前端逆向思路

前言 js逆向一直没有相关了解&#xff0c;虽然目前渗透遇见的不是很多&#xff0c;大多数遇见的要么不加密&#xff0c;要么无法实现其加密流程&#xff0c;不过最近看到了一个较为简单的站点正好能够逆向出来&#xff0c;就做了简单记录。本文旨在介绍js逆向的一些基础思路&a…

C++-详解智能指针

目录 ​编辑 一.什么是智能指针 1.RAII 2.智能智能指针 二.为什么需要智能指针 1.内存泄漏 a. 什么是内存泄漏&#xff0c;内存泄漏的危害 b.内存泄漏分类 c.如何检测内存泄漏 d.如何避免内存泄漏 总结一下: 2.为什么需要智能指针以及智能指针的原理 三.智能指针的使用 1.C…

Leetcode—1657.确定两个字符串是否接近【中等】

2023每日刷题&#xff08;四十五&#xff09; Leetcode—1657.确定两个字符串是否接近 算法思想 源于灵神 实现代码 class Solution { public:bool closeStrings(string word1, string word2) {int len1 word1.size();int len2 word2.size();if(len1 ! len2) {return fa…

Flutter笔记:获取设备信息

Flutter笔记 获取设备信息 作者&#xff1a;李俊才 &#xff08;jcLee95&#xff09;&#xff1a;https://blog.csdn.net/qq_28550263 邮箱 &#xff1a;291148484163.com 本文地址&#xff1a;https://blog.csdn.net/qq_28550263/article/details/134669785 目 录 1. 概述2. 安…

Protocol handler start failed

背景 上一次启动项目还好好的&#xff0c;关闭项目重新打开时&#xff0c;报错了&#xff01; 报错提示 英文&#xff1a;Protocol handler start failed 翻译&#xff1a;协议处理程序启动失败 原因 端口被其他程用了&#xff0c;导致端口冲突。 解决方案 打开任务管理…