博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
数组 泛型 协变(转载)
阅读量:5239 次
发布时间:2019-06-14

本文共 1221 字,大约阅读时间需要 4 分钟。

1、数组的协变性

数组的协变性(covariant)是指:

如果类Base是类Sub的基类,那么Base[]就是Sub[]的基类。

而泛型是不可变的(invariant),List不会是List的基类,更不会是它的子类。

数组的协变性可能会导致一些错误,比如下面的代码:

public static void main(String[] args) {

Object[] array = new String[10];

 

array[0] = 10;

 

}

它是可以编译通过的,因为数组是协变的,Object[]类型的引用可以指向一个String[]类型的对象

但是运行的时候是会报出如下异常的: 

Exception in thread "main" java.lang.ArrayStoreException: java.lang.Integer

 

但是对于泛型就不会出现这种情况了:

public static void main(String[] args) {

List< Object> list = new ArrayList< String>();

list.add(10);

 

}

这段代码连编译都不能通过。

2、数组的具体化。 

数组是具体化的(reified),而泛型在运行时是被擦除的(erasure)。

数组是在运行时才去判断数组元素的类型约束,

而泛型正好相反,在运行时,泛型的类型信息是会被擦除的,只有编译的时候才会对类型进行强化。

所以上面的例子中,数组的方法会在运行时报出ArrayStoreException,而泛型根本无法通过编译。 

 

3、泛型不是协变的

 

 

虽然将集合看作是数组的抽象会有所帮助,但是数组还有一些集合不具备的特殊性质。

 

 语言中的数组是协变的(covariant),也就是说,如果 Integer扩展了 Number(事实也是如此),那么不仅 Integer是 Number,而且 Integer[]也是 Number[],在要求 Number[]的地方完全可以传递或者赋予 Integer[]。(更正式地说,如果 Number是 Integer的超类型,那么 Number[]也是 Integer[]的超类型)。

您也许认为这一原理同样适用于泛型类型 —— List是 List的超类型,那么可以在需要 List的地方传递 List。不幸的是,情况并非如此。

不允许这样做有一个很充分的理由:

这样做将破坏要提供的类型安全泛型。

如果能够将 List赋给 List。

那么下面的代码就允许将非 Integer的内容放入 List

List li = new ArrayList();

List ln = li; // illegal

ln.add(new Float(3.1415));

转载于:https://www.cnblogs.com/xiaolang8762400/p/7113642.html

你可能感兴趣的文章
51nod1076 (边双连通)
查看>>
Item 9: Avoid Conversion Operators in Your APIs(Effective C#)
查看>>
js去除空格
查看>>
学习Spring Boot:(二十八)Spring Security 权限认证
查看>>
IT学习神器——慕课网App获App Store、Android应用市场重磅推荐
查看>>
Linux网络状态工具ss命令使用详解
查看>>
深入浅出JavaScript(2)—ECMAScript
查看>>
编程珠玑第十一章----排序
查看>>
Face The Right Way POJ - 3276 (开关问题)
查看>>
STEP2——《数据分析:企业的贤内助》重点摘要笔记(六)——数据描述
查看>>
变量的命名规范
查看>>
手机端自动跳转
查看>>
react中进入某个详情页URL路劲参数Id获取问题
查看>>
首届.NET Core开源峰会
查看>>
ViewPager的onPageChangeListener里面的一些方法参数:
查看>>
python pdf转word
查看>>
poj 2182 Lost Cows
查看>>
OpenFlow 交换机与控制器交互步骤
查看>>
java-内存模型
查看>>
文本相似度比较(网页版)
查看>>