java 基础 - 泛型

news2025/1/11 5:52:52

泛型

术语中文含义举例
Parameterized type参数化的类型List
Actual typeparameter实际类型参数String
Generic type泛型类型List
Formal typeparameter形式类型参数 E
Unbounded wildcard type无限制通配符类型List<?>
Raw type原始类型List
Bounded type parameter有限制类型参数
Recursive type bound递归类型限制<T extends Comparable>
Bounded wildcard type有限制通配符类型List<? extends Number>
Generic method泛型方法static List asList(E[] a)
Type token类型令牌String.class

不要使用原始类型(如List)

每一种泛型类型都定义一个原生态类型,例如List对应的原生态类型就是List,他们的存在主要是为了与泛型出现之前的代码兼容。

有了泛型之后,类型声明中可以包含信息,而不是通过注释去提醒:

private final Collection<Stamp> stamps = ....

从这个声明中,编译器知道stamps 集合应该只包含Stamp 实例,错误的插入会生成一个编译时错误消息,提醒具体是哪里出错了:

Test.java:9: error: incompatible types: Coin cannot be converted to Stamp
c.add(new Coin());
^
当从集合中检索元素时,编译器会为你插入不可⻅的强制转换

如果使用诸如List 之类的原始类型,则会丢失类型安全性,但是如果使用参数化类型(例如List)则不会

原始类型List逃避了泛型检查,而参数化类型List 明确地告诉编译器,它能够保存任何类型的对象。

可以将List 传递给List 类型的参数,但不能将其传递给List 类型的参数。泛型有子类型化的规则,List 是原始类型 List 的子类型,但不是参数化类型List 的子类型

为了更直观的说明,给出下面的代码:

public static void main(String[] args) {
        List<String> strings = new ArrayList<>();
        unsafeAdd(strings, Integer.valueOf(42));
        String s = strings.get(0); // Has compiler-generated cast
    }
 
    private static void unsafeAdd(List list, Object o) {
        list.add(o);
    }

如果运行该程序,则当程序尝试调用strings.get(0)的结果(一个Integer)转换为一个String 时,会得到ClassCastException 异常。

如果在unsafeAdd的声明中的原始类型List 替换参数化类型List,则编译器直接就会给出报错信息:

在不确定或不在意集合中元素类型时,可能会用到原始类型。例如编写一个返回两个集合中重复元素个数的程序:

static int numElementsInCommon(Set s1, Set s2) {
   int result = 0;
   for (Object o1 : s1)
	   if (s2.contains(o1))
		result++;
    return result;
}

这种方法使用原始类型,是危险的。安全替代方式是使用无限制通配符类型(unbounded wildcard types)。如果要使用泛型类型,但不知道或关心实际类型参数是什么,则可以使用问号来代替。例如,泛型类型 Set 的无限制通配符类型是Set<?>

static int numElementsInCommon(Set<?> s1, Set<?> s2){…}
“不要使用原始类型”这条规则有几个特例情况:

  1. 必须在类签名(class literals)中使用原始类型

例如List.class,String[].class 和int.class 都是合法的,但List.class 和List<?>.class 不合法

  1. 因为泛型类型信息在运行时被擦除,所以在<?>以外的参数化类型上使用instanceof是非法的
    下面是使用泛型类型的instanceof 运算的示例:
if (o instanceof Set) { // Raw type
	Set<?> s = (Set<?>) o; // Wildcard type
	...
}

一旦确定 o 对象是一个Set,则必须将其转换为通配符Set<?>。这是一个强制转换,所以不会导致编译器警告。

消除非受检的警告

使用泛型编程时,会看到许多编译器警告:

  • 非受检强制转换警告(unchecked cast warning)
  • 非受检方法调用警告
  • 非受检参数化可变参数类型警告(unchecked parameterized vararg type warning)
  • 非受检转换警告(unchecked conversion warning)

有些警告非常难消除,但还是要秉承尽可能消除每一个受检警告的原则,如果不能消除警告,但确信引发警告的代码是类型安全的,那么用@SuppressWarnings(“unchecked”)注解来禁止这条警告。

如果在不止一行的方法或构造函数中使用了@SuppressWarnings(“unchecked”),可以将它移动到一个局部变量的声明中。

将@SuppressWarnings(“unchecked”)注解放在return语句中是不合法的,因为它不是一个声明,也不要把注解放在整个方法上,而是应该声明一个局部变量来保存返回值,在局部变量上面添加注解:

public <T> T[] toArray(T[] a) {
	if (a.length < size) {
		// This cast is correct because the array we're creating
		// is of the same type as the one passed in, which is T[].
		@SuppressWarnings("unchecked") 
		T[] result = (T[]) Arrays.copyOf(elements, size, a.getClass());
		return result;
	}
	System.arraycopy(elements, 0, a, 0, size);
	if (a.length > size)
		a[size] = null;
	return a;
}

每当使用@SuppressWarnings(“unchecked”) 注解时,都要写一下注释,说明为什么这么做是安全的

列表优于数组

数组与泛型有很大的不同:

  1. 数组是协变的(covariant)(如果Sub是Super的子类型,则数组类型Sub[] 是数组类型Super[] 的子类型)

  2. 泛型是不变的(invariant)
    对于任何两种不同的类型Type1 和Type2,List 既不是List 的子类型也不是父类型。

现在有两段代码:

//Throws ArrayStoreException
Object[] objectArray = new Long[1];
objectArray[0] = "I don't fit in"; 

// Incompatible types
List<Object> ol = new ArrayList<Long>(); 
ol.add("I don't fit in");

对于上面两种方法无论哪种方式都会报错,因为不能把一个String 类型放到一个Long 类型容器中,但是用一个数组的话,在运行时才会报错;对于列表,可以在编译时就能发现错误。

  1. 数组是具体化的 (在运行时才知道和强化他们的类型
    就比如上面的代码,将String保存到Long数组中就会得到ArrayStoreException异常)

  2. 泛型在编译时就强化它的类型信息,并在运行时擦除它的元素类型信息

》由于上面这些区别,数组和泛型不能很好地混用,所以new List[],new List,new E[]这些语法都是错误的!在编译时会产生一个泛型数组创建错误。

非法的原因是它不类型安全的

List<String>[] stringLists = new List<String>[1]; 	// (1)
List<Integer> intList = List.of(42); 				// (2)
Object[] objects = stringLists; 					// (3)
objects[0] = intList; 								// (4)
String s = stringLists[0].get(0); 					// (5)
  • 假设第1行创建一个泛型数组是合法的
  • 第2行创建并初始化包含单个元素的List
  • 第3行将List 数组存储到Object数组变量中,这是合法的,因为数组是协变的
  • 第4行将List 存储在Object数组的唯一元素中,这是因为泛型是通过擦除来实现的:List[] 实例是List[],所以这个赋值不会产生ArrayStoreException 异常

我们将一个List 实例存储到一个声明为List 实例的数组中,为了防止这种情况出现,第一行必须报错。

E,List 和List 等在技术上被称为不可具体化的类型,指其运行时表示法包含的信息比它的编译时表示法包含的信息更少。唯一可具体化的参数化类型是无限制的通配符类型,如List<?>等,创建无限制通配符类型的数组是合法的,但并不常用。

当泛型数组创建错误时,最佳解决方案是使用集合类型List 。例如编写一个带有集合的Chooser类和一个方法,方法返回集合中随机选择的一个元素。

数组和泛型有着截然不同的类型规则:

  • 数组是协变且可以具体化的
  • 泛型是不可变的且可以被擦除的

优先考虑泛型

以一个简单的栈类实现为例:

public class Stack {
    private Object[] elements;
    private int size = 0;
    private static final int DEFAULT_INITIAL_CAPACITY = 16;
 
    public Stack(){
        elements = new Object[DEFAULT_INITIAL_CAPACITY];
    }
 
    public void push(Object e){
        ensureCapacity();
        elements[size ++] = e;
    }
 
    public Object pop(){
        if (size == 0)
            throw new EmptyStackException();
        Object result = elements[-- size];
        elements[size] = null;
        return result;
    }
 
    public boolean isEmpty(){
        return size == 0;
    }
 
    private void ensureCapacity() {
        if (elements.length == size)
            elements = Arrays.copyOf(elements,  2 * size + 1);
    }
}

//用相应的类型参数替换所有的Object类型:
public class Stack<E> {
    private E[] elements;
    private int size = 0;
    private static final int DEFAULT_INITIAL_CAPACITY = 16;
 
    public Stack(){
        elements = new E[DEFAULT_INITIAL_CAPACITY];
    }
 
    public void push(E e){
        ensureCapacity();
        elements[size ++] = e;
    }
 
    public E pop(){
        if (size == 0)
            throw new EmptyStackException();
        E result = elements[-- size];
        elements[size] = null;
        return result;
    }
}

这个类产生一个错误:不能创建一个不可具体化类型E的数组
在这里插入图片描述
两个解决方法

  1. 创建一个Object数组,将它转换成泛型数组类型

这里需要确保unckecked cast不会危及程序的安全性:相关的数组(elements)保存在一个private的域中,永远不会返回给客户端或传递给任何其他方法。
由于构造方法只包含未经检查的数组创建,所以在整个构造方法中抑制警告。

  1. 将elements的类型从E[] 更改为Object[]

这样会得到一条不同的错误:
![在这里插入图片描述](https://img-blog.csdnimg.cn/7ec7381ab0534a90998def4bdd1955ff.png在这里插入图片描述
可以把从数组中获取到的元素强制转换为E

 E result = (E)elements[-- size];

上面两个方法,第一个方法可读性更强:数组被声明为E[ ]类型以清晰地表示它只包含E实例;第一个方法更简洁:第一种方法只需在创建数组的时候转换一次,第二种方法每次读取一个数组元素时都需要转换一次。

优先考虑泛型方法

静态工具方法尤其适合于泛型化

编写泛型方法类似于编写泛型类:

public static Set union(Set s1, Set s2) {
	 Set result = new HashSet(s1);
	 result.addAll(s2);
	 return result;
}

上面的会收到警告可以用

public static <E> Set<E> union(Set<E> s1, Set<E> s2) {
	Set<E> result = new HashSet<>(s1);
	result.addAll(s2);
	return result;
}

使用上面的方法也很简单:

    public static void main(String[] args) {
        Set<String> guys = CollUtil.newHashSet("Tom", "Dick", "Harry");
        Set<String> stooges = CollUtil.newHashSet("Larry", "Moe", "Curly");
        Set<String> aflCio = union(guys, stooges);
        System.out.println(aflCio);
    }

上面代码的运行结果会输出:[Moe, Tom, Harry, Larry, Curly, Dick]

union 方法的一个限制是所有三个集合(输入参数和返回值)的类型必须完全相同。通过使用限定通配符类型(bounded wildcard types)(Set<? extends xxx>),可以使该方法更加灵活。

除此之外,还有递归类型限制(recursive type bound)的概念:通过包含类型参数本身的表达式来限制类型参数。

递归类型限制的一个经典用法和Comparable接口有关:

public interface Comparable<T> {
	int compareTo(T o);
}

类型参数T,可以与实现Comparable 的类型的元素进行比较。例如,String 类实现了Comparable,Integer 类实现了Comparable,几乎所有类型都只能与同类型的元素比较。

public static <E extends Comparable<E>> E max(Collection<E> c);

<E extends Comparable > 可以理解为「任何可以与自己比较的类型E」

下面的代码实现了计算最大值的功能:

public static <E extends Comparable<E>> E max(Collection<E> c) {
	if (c.isEmpty())
		throw new IllegalArgumentException("Empty collection");
	
	E result = null;
	
	for (E e : c)
		if (result == null || [e.compareTo(result](http://e.compareTo(result)) > 0)
	 		result = Objects.requireNonNull(e);
	
	return result;
}

利用限定通配符来提升API的灵活性

public class Stack<E> {
	public Stack();
	
	public void push(E e);
	
	public E pop();
	
	public boolean isEmpty();
}

//将多个元素放到栈里:
public void pushAll(Iterable<E> src) {
    for (E e : src)
         push(e);
}

Java提供了一种特殊的参数化类型——限定通配符类型(bounded wildcard type),pushAll的输入参数类型应该是「E的某个子类型的Iterable接口」,用代码表示就是Iterable<? extends E>:

public void pushAll(Iterable<? extends E> src) {
   for (E e : src)
         push(e);
}

Collection 不是Collection 的子类型。popAll的输入参数的类型不应该是「E的集合」,而应该是「E的某个父类型的集合」。如下

public void popAll(Collection<? super E> dst) {
     while (!isEmpty())
         dst.add(pop());
}

为了获得最大的灵活性,对代表生产者和消费者的输入参数使用通配符类型

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

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

相关文章

Vue--》Vue3给数据共享增添的改变

目录 数据共享 父向子共享数据 子向父共享数据 父子组件间数据双向同步 兄弟组件共享数据 后代组件共享数据 使用Vue3的setup函数实现后代数据共享 数据共享 在项目开发中&#xff0c;组件之间的关系分为如下3种&#xff1a;父子关系、兄弟关系、后代关系。 父向子共享…

FPGA学习笔记(十二)IP核之FIFO的学习总结

系列文章目录 一、FPGA学习笔记&#xff08;一&#xff09;入门背景、软件及时钟约束 二、FPGA学习笔记&#xff08;二&#xff09;Verilog语法初步学习(语法篇1) 三、FPGA学习笔记&#xff08;三&#xff09; 流水灯入门FPGA设计流程 四、FPGA学习笔记&#xff08;四&…

机器人/人工智能/就业形势2023

机器人/人工智能/就业形势2022https://blog.csdn.net/ZhangRelay/article/details/124441772机器人人工智能课程需求和就业趋势-2022-https://blog.csdn.net/ZhangRelay/article/details/127087308如上已成往事。2023年如何呢&#xff1f;之前文章都过于简洁&#xff0c;很多朋…

浅谈 MySQL 的 Undo log 日志

undo log 存储在类型为 FIL_PAGE_UNDO_LOG 页中。 可以从系统表空间中分配空间&#xff0c;也可以从 undo tablespace 中分配空间。 FIL_PAGE_UNDO_LOG 类型页主要分为四部分&#xff1a; File Header&#xff0c;所有页都有的结构Undo Page Header TRX_UNDO_PAGE_TYPE&#x…

十一、docker相关问题解决方案

&#x1f33b;&#x1f33b;一、创建tomcat失败报348 问题二、端口监听问题&#xff0c;没安装命令三、非正常关闭电脑导致虚拟机无法启动一、创建tomcat失败报348 问题 创建失败问题&#xff1a; docker: Error response from daemon: OCI runtime create failed: container…

通用vue组件化搜索组件页面

一、组件化封装 1.首先创建一个form文件夹&#xff0c;将搜索框组件的内容全部写在这个里面&#xff0c;然后再在需要的页面直接引入相应的组件即可 2.首先先在goods.vue文件里面写对应的文本数组formItems&#xff0c;将所定义的类型IFormItem引用进去&#xff0c;这个里面写…

coresight(五) rom table

五、 rom table 在一个soc中&#xff0c;有多个coresight组件&#xff0c;但是软件怎么去识别这些coresight组件&#xff0c;去获取这些coresight组件的信息了&#xff1f;这个时候&#xff0c;就需要靠coresight组件中&#xff0c;一个重要的组件&#xff0c;这个组件就是rom …

CMD有哪些有趣的命令?

程序员宝藏库&#xff1a;https://gitee.com/sharetech_lee/CS-Books-Store 用惯Linux和macOS的同学都会对各种各样强大的命令印象深刻&#xff0c;然而再转向Windows时就开始不屑一顾&#xff0c;认为Windows上没有Linux上那些超级便捷好用的命令。 其实&#xff0c;Windows下…

ROS安装及rosdep update问题解决

ROS安装&#xff1a; 参考链接&#xff1a;详细介绍如何在ubuntu20.04中安装ROS系统&#xff0c;以及安装过程中出现的常见错误的解决方法&#xff0c;填坑&#xff01;&#xff01;&#xff01;_慕羽★的博客-CSDN博客_ubuntu20.04安装ros rosdep update问题解决&#xff1a…

Linux Shell 脚本编程基础

Shell是一个命令解释器,它解释由用户输入的命令并且把它们送到内核,不仅如此,Shell有自己的编程语言用于对命令的编辑,它允许用户编写由shell命令组成的程序.Shel编程语言具有普通编程语言的很多特点,比如它也有循环结构和分支控制结构等,用这种编程语言编写的Shell程序与其他应…

Selenium用法详解【窗口表单切换】【JAVA爬虫】

简介本文主要讲解java 代码利用Selenium如何实现控制浏览器进行窗口切换和页面内的不同表单之间的切换操作。切换操作窗口切换在 selenium 操作页面的时候&#xff0c;可能会因为点击某个链接而跳转到一个新的页面&#xff08;打开了一个新标签页&#xff09;&#xff0c;这时候…

电子词典流程图

简易流程&#xff1a; 详细介绍 服务端&#xff08;TCP并发&#xff09; 一.分支线程负责处理客户端发送的信息 1.登陆与注册信息 登陆&#xff08;l&#xff09;;注册&#xff08;e&#xff09; (1)登陆根据接收的用户名&#xff0c;密码在用户注册表中遍历是否符合&#xff…

ORB-SLAM2 --- ORBmatcher::SearchBySim3函数

目录 1.函数作用 2.函数流程 3.函数解析 3.1 准备工作&#xff1a;内参&#xff0c;计算Sim3的逆 3.2 记录已经匹配的特征点 3.3 通过Sim变换&#xff0c;寻找 pKF1 中特征点和 pKF2 中的新的匹配 3.4 通过Sim变换&#xff0c;寻找 pKF2 中特征点和 pKF1 中的新的匹…

Jenkins远程SSH部署SpringBoot项目

1.前置环境 前置环境配置&#xff1a;jdk、maven、git 2.在Jenkins配置git凭据 请查看往期文章&#xff1a; https://blog.csdn.net/RookiexiaoMu_a/article/details/122655272?spm1001.2014.3001.5501 3.安装Publish over SSH插件 4.配置SSH Servers 安装完Publish over…

C++蓝桥杯贪心算法

目录 &#x1f33c;一&#xff0c;1812: [NewOJ Week 5] 排列变换 &#x1f33c;二&#xff0c;1827: [NewOJ Week 8] 升降数字 &#x1f33c;三&#xff0c;剑指offer 10-II 青蛙跳台阶问题 &#x1f33c;四&#xff0c;P1223 排队接水 &#x1f33c;五&#xff0c;P5650…

npm常用命令

目录1. 构建项目2. 安装包3. 查看安装目录4. 卸载包5. 更新包6. 查看已安装的包7. 使用国内npm镜像源8. 使用yarn1. 构建项目 npm init # 全部使用默认配置 npm init --yes2. 安装包 # 全局安装 npm install 包名 -g # 本地安装 npm install 包名 # 一次安装多个 npm install…

ROS移动机器人开发——硬件引脚

我们使用的32开发板为冰达机器人官方的开发板&#xff0c;类型为STM32RCT6。32章节的目的为&#xff0c;将官方所给源码转化为 官方标准库函数来进行使用 —————————————— 需求提出&#xff1a; 电源 1. 3.3V-200ma供电 2. 输入 9-12.6V 5A 控制&#xff1a…

(十三)JAVA基础语法

目录 前言: 一、包 二、权限修饰符 三、final关键字 四、常量 五、枚举 六、抽象类 七、抽象类:模板方法模式 八、接口 前言: ①包: 在编写Java程序时&#xff0c;随着程序架构越来越大&#xff0c;类的个数也越来越多&#xff0c;这时就会发现管理程序中维护类名称也…

利用OpenCV的函数calcHist()计算出图像的直方图数据后绘制图像的直方图

利用OpenCV的函数calcHist()计算出图像的直方图数据后绘制图像的直方图 在上一篇博文 https://www.hhai.cc/thread-200-1-1.html 中已经对OpenCV的直方图计算函数calcHist()进行了详细介绍。 这篇博文介绍如何用直方图数据绘制直方图。 OpenCV是没有统计图绘制的相关函数的&a…

Verilog语法笔记(夏宇闻第三版)-循环语句

目录 forever语句: repeat语句: while语句: for语句: 在Verilog HDL中存在着四种类型的循环语句&#xff0c;用来控制执行语句的执行次数。 1) forever 连续的执行语句。 2) repeat 连续执行一条语句 n 次。 3) while 执行一条语句直到某个条件不满足。如果一开始条件即…