解释一下泛型擦除?为什么java必须强制?

news2025/1/4 15:02:31

一、概述:

在解释什么是泛型擦除之前我们得先了解什么是Java泛型。所谓的泛型就是参数化的类型。这就意思着我们可以具体的类型作为一个参数传递给方法、类、接口。

为什么我们需要泛型呢?首先我们都知道在java里,Object就是对象的父类。Object可以引用任何类型的对象。但是这一点会带来类型安全的问题。而泛型的出现就给java带来了类型安全这一项功能。

1、使用泛型有什么好处?

  • 保证类型安全,进行编译期错误检查,使代码具有更好的安全性和可读性。
  • 不需要进行类型强制转换。

如下程序,如果不使用泛型前,强转结果很容易出错:

public class GenericClient {

    public static void main(String[] args) {
        // 不使用泛型,集合中传入的值类型不会受到限制
        // 存值没有限制,取值就容易出错,容器安全性得不到保障
        List list = new ArrayList();
        list.add("tom");
        list.add(11);
        for(Object obj : list){
            // 数据取出,需要进行强转,代码可读性和使用性降低
            String str = (String) obj;
            System.out.println(str);
        }
    }
}

出现了类型转换错误:

tom
Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
	at com.koo.generic.GenericClient.main(GenericClient.java:22)

如果使用泛型,编译器会进行检查,从而规避已知的类型转换异常:
在这里插入图片描述

2、泛型怎么使用

2.1 在类上

class Test<T> {
	T obj
	Test(T obj){
		this.obj = obj;
	}
}

2.2 在方法上

class Test {
	static <T> void helloworld(T t){}
}

2.3 在接口上

interface Test<T> {
	T getData();
}

二、什么是类型擦除?

1、概念

Java 泛型的实现是在编译层,编译后生成的字节码中不包含泛型中的类型信息。所以使用泛型时,加上的类型参数,会在编译器编译的时候去掉,这个过程称为类型擦除。

2、擦除示例

public static <T> boolean myequal(T t1, T t2){
	return t1.equals(t2);
}

编译器会用Object替换掉类型T,如下:

public static <Object> boolean myequal(Object t1, Object t2){
	return t1.equals(t2);
}

3、泛型类的类型擦除

在类级别的类型擦除遵循这样的规则:首先编译器丢弃类上的类型参数,并用它的第一个绑定类型替换它,如果类型参数没有绑定,就用Object来替换。

  • 参数类型没有绑定
public class MyClass<T> {
	private T[] elements;
	public void doSomething(T element){}
	public T getSomething(){}
}

MyClass的类型参数T没有绑定到任何类型,所以将会用Object来替换掉T,替换结果:

public class MyClass {
	private Object[] elements;
	public void doSomething(Object element){}
	public Object getSomething(){}
}
  • 参数类型有绑定
interface MyT {}
public class MyClass<T extends MyT> {
	private E[] elements;
	public void doSomething(T element){}
	public T getSomething(){}
}

MyTClass是MyClass的类型参数T第一个绑定到的类型,因此T将会被替换成MyTClass:

public class MyClass {
	private MyT[] elements;
	public void doSomething(MyT element){}
	public MyT getSomething(){}
}

为什么取第一个绑定就OK了呢?比如说,如果MyT还有父类,父类还有父类,那么我们的类型参数就有了很多间接的绑定,而第一个绑定就覆盖了所有的父类,因此用第一个绑定就可以了。

4、泛型方法的类型擦除

对于泛型方法,它的类型参数不会被存放起来,它遵循这样的规则:首先编译器丢弃方法上的类型参数,并用它的第一个绑定类型替换它,如果类型参数没有绑定,就用Object来替换。

  • 参数类型没有绑定
public static <T> void printSomething(T[] arr){
	for(T item: arr) {
		System.out.printf("%s", item);
	}
}

上面的方法,进行类型擦除的结果后:

public static void printSomething(Object[] arr){
	for(Object item: arr) {
		System.out.printf("%s", item);
	}
}
  • 参数类型有绑定
public static <T extends MyT> void printSomething(T[] arr){
	for(T item: arr) {
		System.out.printf("%s", item);
	}
}

上面的方法,进行类型擦除的结果后:

public static void printSomething(MyT[] arr){
	for(MyT item: arr) {
		System.out.printf("%s", item);
	}
}

三、类型擦除带来的问题

问题描述:

类型信息被擦除,怎么能保证只使用泛型变量的限定类型?编译后 String是Object,Integer 也是 Object,怎么能确定使用哪一个呢?

1、检查针对引用,而非引用对象

什么是引用?
       比如 A a = new A();
       此时变量a指向了一个A对象,a被称为引用变量,也可以说a是A对象的一个引用。我们通过操纵引用变量a来操作A对象。变量a的值为它所引用对象的地址。

为确保正确的使用类型,java的实现顺序是这样的:
先检查泛型类型(针对引用)——类型擦除——编译

如图:
在这里插入图片描述
类型检查就是编译时完成的。new ArrayList()只是在内存中开辟了一个存储空间,可以存储任何的类型对象。而真正涉及类型检查的是它的引用,因为我们是使用它的引用来调用它的方法,比如说list1调用add()方法,它做了泛型限定,所以list1引用能完成泛型类型的检查。而引用list2没有使用泛型,所以没有进行类型检查。

通过上边的例子,我们可以明白,类型检查就是针对引用的。谁是一个引用,用这个引用调用泛型方法,就会针对这个引用调用的方法进行类型检测,而无关它真正引用的对象。

2、泛型中参数化类型不考虑继承关系

下边情况的引用传递是不被允许的——集合不存在类型之间的继承关系
在这里插入图片描述

3、泛型类型变量不能是基本数据类型

不能用类型参数替换基本类型。就比如,没有ArraryList,只有ArraryList。因为当类型擦除后,ArraryList的原始类型变为Object,但是Object类型不能存储double值,只能引用Double的值。

4、运行时类型检查异常——instanceof

例如:

  ArrayList<String> arrayList = new ArrayList<String>();

因为类型擦除后,ArrayList只剩下原始类型,泛型信息String不存在了。
所以你不能这样去判断:

if(arrayList instanceof ArrayList<String>){}

正确的做法是通过通配符的方式:

if(arrayList instanceof ArrayList<?>){}

源码下载:
https://gitee.com/charlinchenlin/koo-erp

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

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

相关文章

2023 年10款「会议管理」软件对比

在这篇文章中&#xff0c;我们将介绍2023年你可以尝试的10个会议管理软件。我们还将介绍会议管理的基础知识&#xff0c;涉及的步骤&#xff0c;以及如何创建会议议程。 2023 年 10 款会议管理软件对比 会议管理软件解决方案是一种数字工具&#xff0c;可以让个人和团队计划、…

一、【Pytorch笔记】pytorch简介,开发环境安装,搭建

一、下载Anaconda Anaconda包括Conda、Python以及一大堆安装好的工具包&#xff08;其他深度学习里用到的东西&#xff09;&#xff0c;比如&#xff1a;numpy、pandas等。所以下了Anaconda就不用下载Python进入Anaconda官网&#xff1a;https://www.anaconda.com/download/下…

【数字 IC】从底层重新认识 D 触发器、建立时间和保持时间

目录 1. NMOS 和 PMOS 2. MOS 管搭建逻辑门 3. 锁存器和触发器 3.1 交叉耦合反相器 3.2 SR 锁存器 3.3 D 锁存器 3.4 D 触发器 4. D 触发器的建立、保持时间 1. NMOS 和 PMOS MOSFET&#xff08;金属氧化物半导体场效应晶体管&#xff09;或 IGFET&#xff08;绝缘栅场…

Web3中文|又一巨头入局?亚马逊或将正式进军NFT领域

行业的风向总随着巨头公司而动。 自从亚马逊公司推出 NFT 市场的消息传出后&#xff0c;大众的目光就锁定在了亚马逊的身上&#xff0c;作为科技巨头的亚马逊入局 NFT 对整个行业都有着举足轻重的意义。 受 FTX 的崩溃的影响&#xff0c;亚马逊 NFT 市场的推出几经波折&#xf…

InnoDB和MySAM有什么区别?

首先&#xff0c;MySAM和InnoDB都是mysql里面的两个存储引擎&#xff0c;mysql5.5版本之前的存储引擎默认是MySAM&#xff0c;mysql5.5版本以后的存储引擎默认是InnoDB,它们底层数据结构都是基于B树的。 Mylsam存储引擎&#xff1a; Mylsam索引是非聚簇索引&#xff0c;Mylsa…

旅游景点剧本杀小程序

旅游景点剧本杀小程序是一种基于微信、支付宝等平台的应用程序&#xff0c;其主要功能包括以下几个方面&#xff1a; 游戏规则和背景介绍&#xff1a;小程序可以提供游戏规则和背景介绍&#xff0c;让玩家更好地了解游戏内容和任务。 队伍组建和角色选择&#xff1a;在…

API网关简介|TaobaoAPI接入

API网关是什么 在日常工作中&#xff0c;不同的场合下&#xff0c;我们可能听说过很多次网关这个名称&#xff0c;这里说的网关特指API网关&#xff08;API Gataway&#xff09;。字面意思是指将所有API的调用统一接入API网关层&#xff0c;由网关层负责接入和输出。 那么在什…

智慧安监方案:AI及视频技术在企业安全生产中的风险预警作用

安全生产是企业生产管理中十分重要部分&#xff0c;在实际场景中&#xff0c;很多作业人员普遍存在安全意识薄弱的问题&#xff0c;大大增加了作业风险。如何建立健全的安全风险研判与监测预警机制&#xff0c;引入大数据、视频识别等技术手段&#xff0c;对安全生产领域潜在风…

EasyCVR视频融合平台设备接入的步骤及端口配置的相关注意事项

EasyCVR视频融合平台基于云边端架构&#xff0c;能实现视频汇聚与集中管理、视频多端分发、多屏展示。平台可支持多协议与多类型设备接入&#xff0c;具体包括国标GB28181、RTMP、RTSP/Onvif、海康Ehome、海康SDK、大华SDK、宇视SDK等&#xff0c;能对外分发RTMP、RTSP、HTTP-F…

SYSU程设c++(第十二周)派生的类型兼容性、对象的类型转换、多重继承、虚基类

派生的类型兼容性 1.可以将派生类的对象赋值给基类对象&#xff0c;反之不可 2.可以将公有派生类对象赋值给基类指针&#xff0c;反之不可 &#xff08;该点必须是公有派生类才兼容&#xff09; 即基类可以被派生类赋值&#xff0c;基类兼容派生类&#xff0c;派生类不兼容基…

采购系统平台在选型中需要考虑那几点?

对于一家企业来说&#xff0c;上线一套系统&#xff0c;不仅仅意味着将业务实现了线上化&#xff0c;还有流程&#xff0c;规则&#xff0c;甚至是人的行为习惯都要随之改变&#xff0c;今天小编和大家一起聊聊采购系统平台选型&#xff0c;需要关注的几点。 1.系统和自身需求的…

图像常见格式及转换(BGR,YUV,NV12,YUV444)

常见格式 RGB RGB 是最常用于显示器的色彩空间&#xff0c;R(red)是红色通道&#xff0c;G(green)是绿色&#xff0c;B(blue)是蓝色通道。这三种颜色以不同的量进行叠加&#xff0c;就可以显示出五彩缤纷的色彩。RGB 格式里(0,0,0)代表着黑色&#xff0c;(255,255,255)代表着…

Unittest自动化测试之单元框架unittest忽略用例

忽略用例 在执行测试脚本的时候&#xff0c;可能会有某几条用例本次不想执行&#xff0c;但又不想删也 不想注释&#xff0c;unittest通过忽略部分测试用例不执行的方式&#xff0c;分无条件忽略和有条 件忽略,通过装饰器实现所描述的场景。 提供的装饰器如下&#xff1a; u…

迅为RK3588开发板打包update.img怎么做

在 uboot kernel android 都编译完成的情况下&#xff0c;才可以打包 update.img&#xff0c;所以一般在完整编 译的时候用。输入以下命令&#xff1a; ./build.sh -u 打包完成会在 rockdev/Image-rk3588_s 目录下生成 update.img 镜像。 iTOP-3588开发板采用瑞芯微RK3588处…

HTTP消息结构

目录 2、客户端请求消息 3、服务器响应消息 4、消息结构&#xff08;同上&#xff09; 5、实例&#xff08;GET&#xff09; 6、HTTP的8种请求方法 7、HTTP中Get和Post 比较 HTTP是基于客户端/服务端&#xff08;C/S&#xff09;的架构模型&#xff0c;通过一个可靠的链接来交换…

科大讯飞AI大模型实测——使用讯飞“星火”设计团队组织架构和工作流程-是骡子是马拉出来溜溜~对比ChatGPT实测体验来啦

讯飞星火是什么 讯飞星火是科大讯飞研发的认知智能大模型&#xff0c;可以和人类进行自然交流&#xff0c;解答问题&#xff0c;高效完成各领域认知智能需求。5月6日下午&#xff0c;科大讯飞召开的“讯飞星火认知大模型”成果发布会上&#xff0c;科大讯飞董事长刘庆峰表示&am…

springcloud项目,localhost:port 可以访问,但是外网IP:port不行

springcloud项目启动后&#xff0c;localhost:port 可以访问&#xff0c;但是外网IP:port不行 现象原因及解决方法解决方法&#xff1a;规则1&#xff1a;规则2&#xff1a;规则3&#xff1a; 现象 springboot项目&#xff0c;localhost:port 可以访问&#xff0c;但是外网IP:…

差分信号知识点

概念 关于差分信号&#xff0c;简单的说就是两根线同时传输信号&#xff0c;振幅相等&#xff0c;极性相反。 优点 1、抗干扰能力强。一般的噪声干扰施加到差分信号上&#xff0c;由于两根线的噪声基本一致&#xff0c;差值仍不变&#xff0c;不会影响到逻辑信号。 2、可以有效…

音视频八股文(7)-- 音频aac adts三层结构

AAC介绍 AAC&#xff08;Advanced Audio Coding&#xff09;是一种现代的音频编码技术&#xff0c;用于数字音频的传输和存储领域。AAC是MPEG-2和MPEG-4标准中的一部分&#xff0c;可提供更高质量的音频数据&#xff0c;并且相比于MP3等旧有音频格式&#xff0c;AAC需要更少的…

【webrtc】编译webrtc

目录 编译webRTC1.visual studio2019 commity安装2.安装depot_tools3.安装ninja4.设置一下网络5.glicent6.拉代码7.切到我们要的分支上8.编译 编译mediasoup1.安装cmake2.编译libmediasoupclient3.生成mediasoup3.1一些报错 3.5 生成socketIO3.6 生成sdptransform4.生成rtn-win…