Effective Java学习笔记第27、28条原生态类型和非受检警告
目录
什么是泛型
泛型与编译器
不要轻易使用原生态类型
可以通过通配符类型来替代原生态类型
几个适合原生态类型的场景
消除非受检的警告
什么是非受检警告
如果无法消除警告
本书27-33条主要介绍泛型。首先介绍什么是泛型,它的应用场景是什么。然后重点介绍27和28条对于泛型编程的两点意见。一是不要轻易使用原生态类型,二是尽量消除非受检警告。
什么是泛型
顾名思义“泛型”指的是“多种类型”,声明中具有一种或多种类型参数的类或者接口成为泛型类或者泛型接口,这两者统称为泛型。常见的泛型接口包括List<T>和Set<T>。
泛型与编译器
首先要明确一个概念,泛型的受用范围仅限于编译阶段,编译完成后通过类型擦除进入执行的程序是不带有泛型信息的,下面举一个例子:
//定义一个简单的泛型
public class Myclass<T> {private T t;public void setT(T t) {this.t = t;}public T getT() {return t;}
}
通过javac将Myclass类编译后,由于类型擦除的原因当你将其反编译后相关的泛型信息会自动被替换成对应的实际类型,以下是通过javap反编译后的内容:
public class Myclass<T> {public Myclass();Code:0: aload_01: invokespecial #1 // Method java/lang/Object."<init>":()V4: returnpublic void setT(T);Code:0: aload_01: aload_12: putfield #7 // Field t:Ljava/lang/Object;5: returnpublic T getT();Code:0: aload_01: getfield #7 // Field t:Ljava/lang/Object;4: areturn
}
这里可以看到,泛型类中的两个泛型方法public void setT(T)和public T get()的输入和输出域原来的泛型T表示都被改换成了Ljava/lang/Object;(这里有的朋友会问为什么从还原的源码中仍然能够看到泛型,那是因为尽管类型擦除移除了泛型的具体类型信息,Java 编译器仍然会在字节码中添加一些额外的信息来支持运行时的类型安全性和泛型的正确使用。而反编译器会利用这些标注自动还原出原有的泛型信息)。
那么在编译阶段编译器会对于泛型做哪些操作来确保类型的安全性和正确性呢?
首先是类型检查:一是类型的兼容性检查,编译器会检查泛型类型的使用是否符合类型兼容性规则。例如,确保泛型参数的使用与上下文中的类型相匹配。二是类型参数限制,如果泛型参数有类型界限(如 T extends Number
),编译器会确保传递给泛型参数的实际类型符合这些界限。
其次是类型推断:编译器能够自动推断出泛型参数的具体类型,这使得开发者在某些情况下不必显式指定泛型参数的类型。比如对于泛型的初始化:
List<String> stringlist = new ArrayList<>();
这里ArrayList<>的初始化就不需要指定对应类型,编译器会自动推断。
类型擦除:编译器在编译阶段会去除泛型类型信息,将泛型类型转换为其对应的原始类型(例如,List<T>
变为 List
)。
插入类型转换:为了确保类型安全,编译器会在适当的位置插入必要的类型转换语句。比如要在一个List<Object>中插入一个Integer,编译器就会讲Integer类型强制转换成Object。
了解了编译器对于泛型进行的各种操作,就可以很好的理解书中的27、28两条。
不要轻易使用原生态类型
原生态类型即隐去了泛型参数信息的类或者接口,比如List是List<String>的原生态类型。这里需要强调的是原生态类型是对应泛型的超类。
很好理解,使用原生态类型以后,类或者接口失去了泛型的属性,自然而然就会绕过编译器的各种类型检测功能,也就失去了泛型在安全性和描述性上所有的优势。也就是说如果程序错误插入了一个不同类型的元素,这个错误会一直保留到运行中当调用这个元素的方法出现问题的时候才会暴露问题;另一方面,编译器无法进行自动的类型转换,因此如果有相关的需求需要靠手动转换。
下面给一个使用了原生态类型的案例,这个原本计划只有Integer的List中混入了String类型,虽然编译器会给出Application.java uses unchecked or unsafe operations.的告警,但是并未显示任何实质性的警告,只有当程序运行中调用到“30”这个元素时才会真正显示错误;另一方面由于编译器未进行插入类型转换,因此对于每一个元素都要进行Integer的强制类型转换,否则系统默认会当作Object来处理。
public class Application {public static void main(String[] args) {List list = new ArrayList();list.add(21);list.add(23);list.add("30");for(Iterator i = list.iterator(); i.hasNext();){System.out.println((Integer)i.next()+1);}}
}
//编译警告
(base) chapter5$ javac Application.java
Note: Application.java uses unchecked or unsafe operations.
Note: Recompile with -Xlint:unchecked for details.//运行异常
(base) chapter5$ java Application
22
24
Exception in thread "main" java.lang.ClassCastException: class java.lang.String cannot be cast to class java.lang.Integer (java.lang.String and java.lang.Integer are in module java.base of loader 'bootstrap')
可以通过通配符类型来替代原生态类型
我们已经知道了原生态类型是有缺陷的,那当需要表示无限制类型的泛型时要如果实现呢?答案是无限制的通配符类型。
通配符类型是泛型功能的扩展,泛型将类和接口的应用类型设置为参数,而通配符就是允许把这个参数设置为一个集合。
public class WildcardDemo {public static void main(String[] args) {List<Integer> intList = new ArrayList<>();List<Number> numList = new ArrayList<>();// 使用上界通配符printNumbers(intList);printNumbers(numList);// 使用下界通配符addElement(intList, 100); // intList 是 Number 的子类型addElement(numList, 100); // numList 本身就是 Number 类型}public static void printNumbers(List<? extends Number> list) {for (Number n : list) {System.out.println(n);}}public static <T extends Comparable<T>> void addElement(List<? super T> list, T element) {list.add(element);}
}
这里printNumbers方法的输入参数是泛型接口List的上界通配符类型,代表该方法可以输入的泛型的类型参数可以是Number的子类。
这里要注意对于无限制通配符类型<?>,不能将任何元素(除null以外)放入,因为编译器无法确定相关元素是否符合类型要求(对于迭代器来说泛型的类型要求是一个未知类型)。
几个适合原生态类型的场景
在类文字中使用原生态类型:类文字是表示类的 Class
对象的一种特殊文字。List.class是合法的,而List<?>.class不合法。
通过instanceof操作符判断类型:用参数化类型使用instanceof操作符是非法的,但是无限制性通配符和原生态类型都是合法的。
消除非受检的警告
什么是非受检警告
首先警告是编译器在编译过程中发现的“异常”,编译过程中会有警告和错误两种“异常”,其中警告是对于运行过程中可能存在问题的提醒,但不影响编译的正常进行,而一旦存在错误将会提前终止编译。下面总结一下编译器的“异常”和程序运行过程中的“异常”。
非受检警告如果被无视就有可能在程序运行中出现各种异常或者错误情况,所以这也是为什么非受检异常无必要消除的原因(另外的弃用警告和未使用变量警告的影响会小一些,但最好还是也注意消除一下)。
如果无法消除警告
如果无法消除,同时证明引起警告的代码是类型安全的,才可以使用一个@SupressWarnings(“unchecked”)注解来禁止这条警告。
@SupressWarnings(“unchecked”)注解可以用在任何粒度的级别中,从单独的局部变量声明到整个类都可以,但是请尽量在尽可能小的范围内使用,否则可能会掩盖重要的警告(永远不要直接在类上使用)。
注意每次使用该注解时,都要添加一条注释,说明这么做为什么是安全的。
相关文章:

Effective Java学习笔记第27、28条原生态类型和非受检警告
目录 什么是泛型 泛型与编译器 不要轻易使用原生态类型 可以通过通配符类型来替代原生态类型 几个适合原生态类型的场景 消除非受检的警告 什么是非受检警告 如果无法消除警告 本书27-33条主要介绍泛型。首先介绍什么是泛型,它的应用场景是什么。然后重点介…...
javaEE和javaSE
引用自:https://developer.baidu.com/article/detail.html?id3312755 文章目录 前景描述javaSE简介使用场景 javaEE(J2EE)简介使用场景 结语 前景描述 javaEE和javaSE是java中比较常见的两个概念,但是又比较容易忘记,在此进行记…...

Leetcode 17.电话号码的字母组合
目录 题目 方法一 思路 代码 题目 17. 电话号码的字母组合 难度:中等 给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。 给出数字到字母的映射如下(与电话按键相同)。注意 1 不对…...

位1的个数
编写一个函数,获取一个正整数的二进制形式并返回其二进制表达式中设置位的个数(也被称为汉明重量)。 示例 1: 输入:n 11 输出:3 解释:输入的二进制串 1011 中,共有 3 个设置位。示…...

RPA在政务服务中的挑战与解决方案
随着数字化时代的到来,数字政务的建设已成必然趋势,RPA作为数字化转型的重要工具之一,能够帮助政府单位快速实现业务流程的自动化和智能化,提高工作效率和质量,为建设数字政务提供强有力的支持,因此正被越来…...

RabbitMQ docker安装
后台配置文件 rabbitmq:image: rabbitmq:latestcontainer_name: rabbitmqports:- "5672:5672" # RabbitMQ server port- "15672:15672" # RabbitMQ management console portenvironment:RABBITMQ_DEFAULT_USER: adminRABBITMQ_DEFAULT_PASS: admin 若要打…...

关于vs调试的一些基本技巧方法,建议新手学习
文章目录 1.Debug 和 Release2.VS的调试快捷键3.对程序的监视和内存观察3.1监视3.2内存 4.编程常见错误归类4.1编译型错误4.2链接型错误4.3运行时错误 1.Debug 和 Release 在我们使用的编译器 vs 中,这个位置有两个选项,分别为Debug和Release,…...
MySQL——索引(二)创建索引(2)使用 CREATE INDEX 语句在已经存在的表上创建索引
若想在一个已经存在的表上创建索引,可以使用 CREATE INDEX 语句,CREATEINDEX语句创建索引的具体语法格式如下所示: CREATE [UNIQUEIFULLTEXTISPATIAL]INDEX 索引名 ON 表名(字段名[(长度)J[ASCIDESC]); 在上述语法格式中,UNIQUE、FULLTEXT 和…...
前端HTML总结
目录 前言 正文 head SEO body 网页的主要组成元素: body标签中常见的标签: 自闭合标签: 无语义标签: 特殊符号: 列表 子项: 样式修改: 定义列表: 语义化࿱…...
【动态规划】647. 回文子串
力扣链接:. - 力扣(LeetCode) 动规大法开始吟唱: dp[i][j]含义:从i到j的子串是否为回文子串 递推公式:当s[i] s[j]时 1. j-i<1时, dp[i][j]为true 2. 否则,若dp[i1][j-1]为true&#x…...

python-约瑟夫环(赛氪OJ)
[题目描述] n 个人( 0,1,2,3,4...n−1 ),围成一圈,从编号为 k 的人开始报数,报数报到 m 的人出队。 下次从出队的人之后开始重新报数,循环往复,当队伍中只剩最后一个人的时候,那个人…...
Less 教程:从入门到精通
Less 教程:从入门到精通 1. 引言 Less 是一种流行的动态样式表语言,它扩展了 CSS 的功能,使其更加强大和灵活。通过本教程,我们将深入探讨 Less 的基本概念、特性以及如何在项目中实际应用它。 2. Less 的基本概念 2.1 变量 …...

【VScode】如何在anaconda虚拟环境中打开vscode项目
文章目录 【必备知识】打开anaconda虚拟环境切换到项目工作目录激活anaconda虚拟路径让vscode从当前目录打开 【必备知识】 anaconda环境变量配置及配置python虚拟环境 https://blog.csdn.net/xzzteach/article/details/140621596 打开anaconda虚拟环境 切换到项目工作目录 …...

Flink任务提交流程和运行模式
任务提交流程 Flink 的提交流程随着部署模式、资源管理平台的不同,会有不同的变化。这里做进一步的抽象,形成一个大概高视角的任务执行流程图,如下: Flink按照集群和资源管理的划分运行模式有:Standalone、Flink On…...

【机器学习】 Sigmoid函数:机器学习中的关键激活函数
🌈个人主页: 鑫宝Code 🔥热门专栏: 闲话杂谈| 炫酷HTML | JavaScript基础 💫个人格言: "如无必要,勿增实体" 文章目录 Sigmoid函数:机器学习中的关键激活函数1. 引言2. Sigmoid函数定义3.…...

Vue+Element Plus后台管理主界面搭建实现
续接Django REST Framework,使用Vite构建Vue3的前端项目 1. 后台管理系统主界面框架搭建 后台系统主界面搭建 新建后台管理文件目录 完成后台整体布局 // 1.主界面 index.vue<script setup lang"ts"></script><template><el-…...

JAVA—异常
认识异常,学会从报错信息中发现问题,解决问题。并学会构建自定义异常,提醒编程时注意 目录 1.认识异常 2.自定义异常 1.自定义运行时异常 2.自定义编译时异常 3.异常的处理 1.认识异常 异常就是代表程序出现的问题,用来查询B…...

常见八股面试题:Dubbo 和 Spring Cloud Gateway 有什么区别?
大家好,我是鸭鸭! 此答案节选自鸭鸭最近弄的面试刷题神器面试鸭,更多大厂常问面试题,可以点击进行阅读哈! 目前这个面试刷题神器刚出,有网页和小程序双端可以阅读! 回归面试题! …...

k8s分布式存储-ceph
文章目录 Cephdeploy-ceph部署1.系统环境初始化1.1 修改主机名,DNS解析1.2 时间同步1.3 配置apt基础源与ceph源1.4关闭selinux与防火墙1.5 **创建** ceph **集群部署用户** cephadmin1.6分发密钥 2. ceph部署2.1 **安装** ceph 部署工具2.2 **初始化** mon **节点**…...
Redis cluster集群部署
redis搭建集群模式、Cluster模式(6节点,3主3从集群模式,添加删除节点)_redis cluster节点带数据增减-CSDN博客...

【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...

Linux --进程控制
本文从以下五个方面来初步认识进程控制: 目录 进程创建 进程终止 进程等待 进程替换 模拟实现一个微型shell 进程创建 在Linux系统中我们可以在一个进程使用系统调用fork()来创建子进程,创建出来的进程就是子进程,原来的进程为父进程。…...
Linux离线(zip方式)安装docker
目录 基础信息操作系统信息docker信息 安装实例安装步骤示例 遇到的问题问题1:修改默认工作路径启动失败问题2 找不到对应组 基础信息 操作系统信息 OS版本:CentOS 7 64位 内核版本:3.10.0 相关命令: uname -rcat /etc/os-rele…...

初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...

【C++特殊工具与技术】优化内存分配(一):C++中的内存分配
目录 一、C 内存的基本概念 1.1 内存的物理与逻辑结构 1.2 C 程序的内存区域划分 二、栈内存分配 2.1 栈内存的特点 2.2 栈内存分配示例 三、堆内存分配 3.1 new和delete操作符 4.2 内存泄漏与悬空指针问题 4.3 new和delete的重载 四、智能指针…...

基于Springboot+Vue的办公管理系统
角色: 管理员、员工 技术: 后端: SpringBoot, Vue2, MySQL, Mybatis-Plus 前端: Vue2, Element-UI, Axios, Echarts, Vue-Router 核心功能: 该办公管理系统是一个综合性的企业内部管理平台,旨在提升企业运营效率和员工管理水…...

华为OD机试-最短木板长度-二分法(A卷,100分)
此题是一个最大化最小值的典型例题, 因为搜索范围是有界的,上界最大木板长度补充的全部木料长度,下界最小木板长度; 即left0,right10^6; 我们可以设置一个候选值x(mid),将木板的长度全部都补充到x,如果成功…...
redis和redission的区别
Redis 和 Redisson 是两个密切相关但又本质不同的技术,它们扮演着完全不同的角色: Redis: 内存数据库/数据结构存储 本质: 它是一个开源的、高性能的、基于内存的 键值存储数据库。它也可以将数据持久化到磁盘。 核心功能: 提供丰…...
Java求职者面试指南:Spring、Spring Boot、Spring MVC与MyBatis技术解析
Java求职者面试指南:Spring、Spring Boot、Spring MVC与MyBatis技术解析 一、第一轮基础概念问题 1. Spring框架的核心容器是什么?它的作用是什么? Spring框架的核心容器是IoC(控制反转)容器。它的主要作用是管理对…...

Python训练营-Day26-函数专题1:函数定义与参数
题目1:计算圆的面积 任务: 编写一个名为 calculate_circle_area 的函数,该函数接收圆的半径 radius 作为参数,并返回圆的面积。圆的面积 π * radius (可以使用 math.pi 作为 π 的值)要求:函数接收一个位置参数 radi…...