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

Java泛型设计详解

引言

在日常Java开发中,泛型是一个非常重要的特性。它提供了编译时的类型安全检查,增强了代码的可读性和可维护性。然而,对于初学者甚至一些有经验的开发者来说,泛型的使用和理解仍然是一个挑战。本文旨在深入探讨Java泛型的诞生背景、特性以及著名的PECS原则,帮助读者更好地掌握这一强大的工具。

泛型的诞生背景

为什么需要泛型

在没有泛型之前,Java中的集合类(如ArrayListHashMap等)只能存储Object类型的对象。这意味着在添加和获取元素时,需要进行显式的类型转换。这种做法不仅繁琐,而且容易导致ClassCastException。例如:

ArrayList list = new ArrayList();
list.add(11);
list.add("ssss");
for (int i = 0; i < list.size(); i++) {System.out.println((String) list.get(i));
}

上述代码在运行时会抛出ClassCastException,因为列表中的元素既有Integer类型,又有String类型。在尝试将Integer类型转换为String类型时,会抛出异常。

为了解决这一问题,Java 5引入了泛型。泛型允许在编译时指定集合中元素的类型,从而在编译阶段就能检查类型错误,避免运行时异常。例如:

List<String> list = new ArrayList<>();
list.add("hahah");
list.add("ssss");
for (int i = 0; i < list.size(); i++) {System.out.println(list.get(i));
}

在上面的代码中,我们只能向列表中添加String类型的元素,否则编译器会报错。这样,在调用get()方法时,无需进行显式的类型转换,因为编译器已经确保了元素的类型。

泛型的思想来源

泛型的思想并不是Java独有的,它在其他编程语言中早已存在。例如,C++中的模板(Templates)就是一种参数化类型的机制。模板允许开发者编写与类型无关的代码,在编译时根据实际的类型参数生成具体的代码。Java的泛型在很大程度上借鉴了C++模板的思想。

泛型的特性

类型安全

泛型最显著的特性之一是类型安全。通过泛型,可以在编译时捕获类型错误,而不是在运行时。这大大提高了程序的稳定性和可靠性。例如:

List<Integer> intList = new ArrayList<>();
intList.add(1);
intList.add(2);
int firstInt = intList.get(0); // 正确,无需类型转换
// intList.add("three"); // 编译错误,无法添加String类型元素

在上面的代码中,尝试向intList中添加String类型的元素会导致编译错误。这是因为编译器在编译时已经检查了元素的类型,并确保了类型的一致性。

代码复用

泛型允许编写适用于多种数据类型的代码,从而避免了代码重复。例如,可以编写一个通用的排序方法,该方法可以对任何实现了Comparable接口的对象进行排序:

java复制代码
public static <T extends Comparable<T>> void sort(List<T> list) {
// 实现排序算法
}

在这个方法中,T是一个类型参数,它表示列表中的元素类型。由于T继承自Comparable<T>,因此可以对列表中的元素进行比较和排序。这样,一个排序方法就可以适用于任何类型的列表,而无需为每种类型编写单独的排序方法。

灵活性

泛型提供了在运行时动态指定类型的能力,从而增加了代码的灵活性。例如,可以编写一个泛型类,该类可以在创建对象时指定类型参数:

public class Box<T> {
private T t;
public void set(T t) {
this.t = t;}
public T get() {
return t;}
}
// 使用Box类
Box<Integer> box1 = new Box<>();
box1.set(12);
Integer value1 = box1.get();
Box<String> box2 = new Box<>();
box2.set("abc");
String value2 = box2.get();

在这个例子中,Box类是一个泛型类,它使用类型参数T来表示存储的元素类型。通过创建Box<Integer>Box<String>对象,可以分别存储整数和字符串类型的元素。

PECS原则的由来

什么是PECS原则

PECS原则是“Producer Extends, Consumer Super”的缩写,它是由Joshua Bloch在《Effective Java》一书中提出的。PECS原则旨在指导开发者在使用Java泛型时的通配符选择,特别是在涉及到集合类时。

  • Producer Extends:当你需要从一个数据结构获取(生产)元素时,应该使用extends通配符。这样,你可以从该数据结构读取到的类型为该通配符或其任何子类型的对象。
  • Consumer Super:当你需要向一个数据结构写入(消费)元素时,应该使用super通配符。这允许你写入指定的类型或其任何超类型(父类型)的对象到该数据结构中。

PECS原则的由来

PECS原则的提出是为了解决在使用泛型通配符时可能出现的类型安全问题。在没有PECS原则指导的情况下,开发者可能会错误地选择通配符,从而导致类型不匹配或类型转换异常。

例如,考虑以下两个方法:

public void printElements(List<? extends Number> list) {
for (Number number : list) {System.out.println(number);}
}
public void addElements(List<? super Integer> list) {list.add(1);list.add(2);
}

printElements方法中,我们使用了extends通配符。这是因为我们只需要从列表中读取元素,而不需要向列表中添加元素。使用extends通配符可以确保我们读取到的元素是Number类型或其子类型,从而避免了类型转换异常。

addElements方法中,我们使用了super通配符。这是因为我们需要向列表中添加元素,而添加的元素类型是Integer。使用super通配符可以确保我们可以将Integer类型或其父类型的对象添加到列表中。

PECS原则的应用实例

使用extends进行读取操作

假设我们有一个Animal类和一个Bird类,其中Bird继承自Animal。现在,我们想要编写一个方法来打印动物列表中所有鸟的名字:

public class Animal {String name;
public Animal(String name) {
this.name = name;}
public String getName() {
return name;}
}
public class Bird extends Animal {
public Bird(String name) {
super(name);}
}
public void printBirdNames(List<? extends Bird> birdList) {
for (Bird bird : birdList) {System.out.println(bird.getName());}
}
// 使用方法
List<Bird> birdList = Arrays.asList(new Bird("Parrot"), new Bird("Sparrow"));
printBirdNames(birdList);

在这个例子中,printBirdNames方法接受一个List<? extends Bird>类型的参数。这意味着我们可以传递一个包含Bird类型或其子类型的列表给该方法。由于我们只从列表中读取元素,因此使用extends通配符是合适的。

使用super进行写入操作

现在,假设我们想要编写一个方法来向动物列表中添加一些默认的鸟:

public void addDefaultBirds(List<? super Bird> animalList) {animalList.add(new Bird("Default Parrot"));animalList.add(new Bird("Default Sparrow"));
}
// 使用方法
List<Animal> animalList = new ArrayList<>();
addDefaultBirds(animalList);

在这个例子中,addDefaultBirds方法接受一个List<? super Bird>类型的参数。这意味着我们可以传递一个List<Animal>List<Bird>List<Object>类型的列表给该方法。由于我们需要向列表中添加元素,并且添加的元素类型是Bird,因此使用super通配符是合适的。

泛型的类型擦除

什么是类型擦除

Java的泛型是在编译时实现的,而不是在运行时。这意味着在编译后生成的字节码中,泛型信息会被擦除。这种机制被称为类型擦除(Type Erasure)。

类型擦除的目的

类型擦除的目的是为了保持与Java 5之前版本的兼容性。由于泛型是在Java 5中引入的,而在此之前已经存在大量的Java代码,因此为了兼容这些代码,Java设计者选择了类型擦除这种方案。

类型擦除的影响

类型擦除对泛型的使用产生了一些限制和影响:

  1. 类型参数不支持基本类型:由于类型擦除会将泛型类型替换为它们的上界(通常是Object),而Object不能存储基本类型的值,因此泛型类型参数不支持基本类型。
  2. 不能实例化类型参数:由于类型擦除,无法在运行时创建泛型类型的实例。例如,无法创建T[]类型的数组。
  3. 不能创建泛型数组的实例:同样由于类型擦除,无法创建泛型数组的实例。例如,new T[10]是非法的。
  4. 运行时类型检查受限:由于泛型信息在运行时被擦除,因此无法使用instanceof关键字来检查泛型类型的实例。例如,if (obj instanceof T)是非法的。

类型擦除的实现原理

在编译时,Java编译器会对泛型代码进行类型擦除处理。具体来说,编译器会将泛型类型替换为它们的上界,并插入必要的类型转换代码以确保类型安全性。

例如,考虑以下泛型方法:

public static <T> T getFirstElement(List<T> list) {
return list.get(0);
}

在编译后,该方法会被转换为类似以下的代码:

public static Object getFirstElement(List list) {
return (T) list.get(0); // 插入类型转换
}

注意,在转换后的代码中,泛型类型T被替换为了Object,并且在返回语句中插入了类型转换。这是为了确保类型安全性,因为编译器知道在调用getFirstElement方法时,传入的列表应该是特定类型的列表(例如List<String>),因此可以将返回的对象强制转换为该类型。

泛型的边界限定

什么是边界限定

边界限定(Bounded Types)允许对泛型类型参数施加约束。通过边界限定,可以确保泛型类型参数是某个特定类或接口的子类型或实现类型。

边界限定的语法

边界限定的语法如下:

java复制代码
<T extends Bound>

其中,T是泛型类型参数,Bound是对T的约束。Bound可以是一个类或接口。

边界限定的应用实例

泛型类的边界限定

假设我们想要编写一个泛型类,该类只能处理数值类型的对象。我们可以使用边界限定来确保这一点:

public class NumericBox<T extends Number> {
private T value;
public void setValue(T value) {
this.value = value;}
public T getValue() {
return value;}
}
// 使用NumericBox类
NumericBox<Integer> intBox = new NumericBox<>();
intBox.setValue(123);
Integer intValue = intBox.getValue();
NumericBox<Double> doubleBox = new NumericBox<>();
doubleBox.setValue(123.45);
Double doubleValue = doubleBox.getValue();

在这个例子中,NumericBox类是一个泛型类,它使用边界限定T extends Number来确保T只能是Number类型或其子类型(如IntegerDouble等)。

泛型方法的边界限定

同样地,我们也可以对泛型方法进行边界限定。例如,我们可以编写一个泛型方法来计算两个数值的和:

public static <T extends Number> double sum(T a, T b) {
return a.doubleValue() + b.doubleValue();
}
// 使用sum方法
double result1 = sum(123, 456);
double result2 = sum(123.45, 67.89);

在这个例子中,sum方法是一个泛型方法,它使用边界限定T extends Number来确保ab只能是Number类型或其子类型。然后,方法将ab转换为double类型并计算它们的和。

总结

Java泛型是一个强大的特性,它提供了编译时的类型安全检查、代码复用和灵活性。通过深入理解泛型的诞生背景、特性以及PECS原则,我们可以更好地利用泛型来编写高质量、可维护的Java代码。同时,我们也需要注意泛型的一些限制和影响,如类型擦除和边界限定等。在实际开发中,合理使用泛型将大大提高代码的质量和效率。

相关文章:

Java泛型设计详解

引言 在日常Java开发中&#xff0c;泛型是一个非常重要的特性。它提供了编译时的类型安全检查&#xff0c;增强了代码的可读性和可维护性。然而&#xff0c;对于初学者甚至一些有经验的开发者来说&#xff0c;泛型的使用和理解仍然是一个挑战。本文旨在深入探讨Java泛型的诞生…...

用ue5打开网址链接

需要用到 Launch URL 这个函数 字面意思就是打开填写的链接网页 这里填写的是百度&#xff0c;按下Tab键后就会打开百度的网页...

【大数据】-- 读放大和写放大

目录 一、定义 1. 读放大(Read Amplification) 定义 原因 优化方法 2. 写放大(Write Amplification) 定义 原因 优化方法 对比与联系 二、举例 1. Hadoop(HDFS) 读放大 写放大 2. Flink 读放大 写放大 3. Hive 读放大 写放大 4. Presto 读放大 写放…...

【前端】JavaScript 抽取字符串特定部分题目详解与实现思路

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: 前端 文章目录 &#x1f4af;前言&#x1f4af;题目描述&#x1f4af;核心步骤与实现解析1. 分割字符串为数组&#xff08;split 方法&#xff09;2. 使用 filter 提取名字&#xff08;偶数索引判断&#xff09;3. 使…...

CNCF云原生生态版图-分类指南(一)- 观测和分析

CNCF云原生生态版图-分类指南&#xff08;一&#xff09;- 观测和分析 CNCF云原生生态版图-分类指南一、观测和分析&#xff08;Observability and Analysis&#xff09;&#xff08;一&#xff09;可观测性&#xff08;Observablility&#xff09;1. 是什么&#xff1f;2. 解决…...

热更新解决方案3 —— xLua

概述 xLua框架导入和AB包相关准备 xLua导入 其它的导入 C#调用Lua 1.Lua解析器 using System.Collections; using System.Collections.Generic; using UnityEngine; //引用命名空间 using XLua;public class Lesson1_LuaEnv : MonoBehaviour {// Start is called before the fi…...

如何让ai在游戏中更像一个人?

开题开了一整年是我没想到的&#xff0c;还因此延毕了……我重新梳理一下我想做的研究以及相关痕迹。 我2023年3月找到的导师。起初我发现了在玩RTS游戏中会出现很多固定的套路&#xff0c;选手为此要做大量的练习&#xff0c;我就在想如何把这部分内容借助状态机这种流程给…...

websocket_asyncio

WebSocket 和 asyncio 指南 简介 本指南涵盖了使用 Python 中的 websockets 库进行 WebSocket 编程的基础知识&#xff0c;以及 asyncio 在异步非阻塞 I/O 中的作用。它提供了构建高效 WebSocket 服务端和客户端的知识&#xff0c;以及 asyncio 的特性和优势。 1. 什么是 WebS…...

如何在NGINX中实现基于IP的访问控制(IP黑白名单)?

大家好&#xff0c;我是锋哥。今天分享关于【如何在NGINX中实现基于IP的访问控制&#xff08;IP黑白名单&#xff09;&#xff1f;】面试题。希望对大家有帮助&#xff1b; 如何在NGINX中实现基于IP的访问控制&#xff08;IP黑白名单&#xff09;&#xff1f; 1000道 互联网大…...

Y3编辑器文档4:触发器1(界面及使用简介、变量作用域、入门案例)

文章目录 一、触发器简介1.1 触发器界面1.2 ECA语句编辑及快捷键1.3 参数设置1.4 变量设置1.5 实体触发器1.6 触发器复用 二、触发器的多层结构2.1 子触发器&#xff08;在游戏内对新的事件进行注册&#xff09;2.2 触发器变量作用域 三、入门案例3.1 使用触发器实现瞬间移动3.…...

echarts图表自定义配置(二)——代码封装

下图是初版&#xff0c;火山图的代码。可以看出&#xff0c;里面的变量&#xff0c;逻辑&#xff0c;函数存在冗余&#xff0c;基本上都是改了参数&#xff0c;同样的get和set&#xff0c;去刷新图表&#xff1b;对于往后继续开发十几二十个图表&#xff0c;会很麻烦。因此需要…...

02、10个富士胶片模拟的设置

二色彩 1、色彩的加减控制全局的饱和度增减&#xff1b; 2、色彩效果只提升暖色系饱和度&#xff1b; 3、FX蓝色大幅度提升蓝色系饱和度&#xff1b; 4、三个参数都不改变颜色的色相。 2.1 色彩 色彩调整的是拍摄画面整体的色彩饱和程度 2.2色彩效果 调整的是画面中暖色…...

鸿蒙系统-前端0帧起手

鸿蒙系统-前端0帧起手 先search 一番 找到对应的入门文档1. 运行项目遇到问题 如下 &#xff08;手动设计npm 的 registry 运行 npm config set registry https://registry.npmjs.org/&#xff09;2.运行后不支持一些模拟器 配置一下&#xff08;如下图&#xff0c;运行成功&am…...

211-基于FMC的1路1.5G ADC 1路 2.5G DAC子卡

一、板卡概述 FMC-1AD-1DA-1SYNC是我司自主研发的一款1路1G AD采集、1路2.5G DA回放的FMC、1路AD同步信号子卡。板卡采用标准FMC子卡架构&#xff0c;可方便地与其他FMC板卡实现高速互联&#xff0c;可广泛用于高频模拟信号采集等领域。 二、功能介绍 2.1 原理框图 2.2 硬件…...

获取微信用户openid

附上开发文档:https://developers.weixin.qq.com/doc/offiaccount/OA_Web_Apps/Wechat_webpage_authorization.html 开发之前,准备事项 一个已认证过的服务号|基本信息配置js域名和网站授权域名配置最后确认当前账号网页授权功能是否开通,没有开通的无法获取到用户授权开发人…...

MultiRECloudSim使用

MultiRECloudSim使用 简介 MultiRECloudSim是一个用于云计算环境下的模拟器相关工具,它主要用于模拟和评估云计算中的资源分配、任务调度等多种场景。它可能是基于CloudSim这个基础的云计算模拟器进行扩展而来,CloudSim提供了基本的云计算模拟功能,如数据中心、虚拟机、任务…...

智能设备安全-固件逆向分析

固件逆向分析实验报告-20241022 使用固件常用逆向分析工具&#xff0c;对提供的固件进行文件系统提取&#xff0c;并记录逆向分析实验过程&#xff0c;提交实验报告&#xff08;报告要求图文并茂&#xff0c;对涉及到的关键步骤附截图说明&#xff09;。具体任务如下&#xff1…...

【小白包会的】使用supervisor 管理docker内多进程

使用supervisor 管理docker内多进程 一般情况下&#xff0c;一个docker是仅仅运行一个服务的 但是有的情况中&#xff0c;希望一个docker中运行多个进程&#xff0c;运行多个服务&#xff0c;也就是一个docker容器执行多个服务。 调研了一下&#xff0c;发现可以通过**super…...

使用navicat新旧版本,连接PostgreSQL高版本报错问题图文解决办法

使用navicat新旧版本&#xff0c;连接PostgreSQL高版本报错问题图文解决办法 一、问题现象&#xff1a;二、出现原因三、解决方法&#xff1a;1、升级Navicat版本&#xff1a;2、使用低版本的postgreSQL&#xff1a;3、修改Navicat的dll二进制文件&#xff1a;navicat版本15nav…...

IDEA 未启用lombok插件的Bug

项目中maven已引用了lombok依赖&#xff0c;之前运行没有问题的&#xff0c;但有时启动会提示&#xff1a; java: You arent using a compiler supported by lombok, so lombok will not work and has been disabled. Your processor is: com.sun.proxy.$Proxy8 Lombok support…...

认识GO--gRPC的metadata

参考&#xff1a; 写给go开发者的gRPC教程-metadata-CSDN博客https://blog.csdn.net/kevin_tech/article/details/129395177?ops_request_misc%257B%2522request%255Fid%2522%253A%25221f2f2e26f48c755c33344ccb171a49fc%2522%252C%2522scm%2522%253A%252220140713.130102334…...

2024年安徽省职业院校技能大赛信息安全管理与评估

一、赛项名称 赛项名称&#xff1a;信息安全管理与评估 英文名称&#xff1a;Information Security Management and Evaluation 赛项组别&#xff1a;高职组 赛项归属&#xff1a;电子信息大类 二、竞赛目标 通过赛项检验参赛选手熟悉信息安全行业标准规范和信息 安全测试员新职…...

Perl 引用

Perl 引用 Perl&#xff0c;作为一种灵活而强大的编程语言&#xff0c;广泛用于系统管理、网络编程、GUI开发等领域。在Perl编程中&#xff0c;引用&#xff08;References&#xff09;是一个核心概念&#xff0c;它允许变量引用其他数据&#xff0c;从而创建复杂的数据结构&a…...

RT-Thread启动过程 :从汇编开始的启动流程

这个系列参考了《嵌入式实时操作系统RT-Thread设计与实现》&#xff0c;会详细介绍RT-Thread的启动流程&#xff0c;即是如何从零开始在开发板上运行起一个RTOS内核的。本文将会以 ch32v307VCT6 开发板为例展开进行详细介绍。主要包括&#xff1a;startup.S、初始化与系统相关的…...

Scala—“==“和“equals“用法(附与Java对比)

Scala 字符串比较—""和"equals"用法 Scala 的 在 Scala 中&#xff0c; 是一个方法调用&#xff0c;实际上等价于调用 equals 方法。不仅适用于字符串&#xff0c;还可以用于任何类型&#xff0c;并且自动处理 null。 Demo&#xff1a; Java 的 在 J…...

$route和$router的区别

在 Vue.js 中&#xff0c;$route 和 $router 都是 Vue Router 提供的对象&#xff0c;但它们有不同的用途和功能。 1. $router $router 是 Vue Router 实例的引用&#xff0c;它允许你通过 JavaScript 进行路由的控制和导航。你可以通过 $router 来执行路由的操作&#xff0c…...

[工具升级问题] 钉钉(linux版)升级带来的小麻烦

本文由Markdown语法编辑器编辑完成。 1. 背景: 今日钉钉又发布了新的升级版本。由于我工作时使用的是Ubuntu 20.04版本&#xff0c;收到的升级推送信息是&#xff0c;可以升级到最新的7.6.25-Release版本。根据钉钉官方给出的历次更新版说明&#xff0c;这个新的版本&#xf…...

Leetcode经典题13--接雨水

题目描述 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图&#xff0c;计算按此排列的柱子&#xff0c;下雨之后能接多少雨水。 输入输出示例 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;6 解释&#xff1a;上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1…...

yarn修改缓存位置

查看缓存位置 以下三个命令分别为&#xff1a;bin是yarn存储命令的二进制文件&#xff0c;global存储全局node_modules &#xff0c;cache存储用下下载缓存&#xff0c;查看本机目前的目录&#xff1a; 查看bin目录命令&#xff1a;yarn global bin 查看global目录命令&…...

OpenHarmony-3.HDF input子系统(5)

HDF input 子系统OpenHarmony-4.0-Release 1.Input 概述 输入设备是用户与计算机系统进行人机交互的主要装置之一&#xff0c;是用户与计算机或者其他设备通信的桥梁。常见的输入设备有键盘、鼠标、游戏杆、触摸屏等。本文档将介绍基于 HDF_Input 模型的触摸屏器件 IC 为 GT91…...