Java泛型中的协变和逆变(哈工大软件构造)

  在软件构造课程的进行中,讲到复用性和LSP原则的时候,提到了一些协变和逆变的概念,以及在特定的数据结构,如 数组Collections泛型 中的协变和逆变情况。但在课堂上并没有完全理解,通过在课后查询资料,有一点收获,在此分享。

协变和逆变?

  简而言之,协变和逆变是用来描述类型转换前后的继承关系的两个概念。以A 、B表示类型,f(⋅)表示类型转换(f(A)表示A转换后的类型),≤表示继承关系(比如,A≤B表示A是由B派生出来的子类),其定义可概括如下:
f(⋅)是协变(covariant)的,当A≤B时有f(A)≤f(B)成立;
f(⋅)是逆变(contravariant)的,当A≤B时有f(B)≤f(A)成立;
f(⋅)是不变(invariant)的,当A≤B时上述两个式子均不成立,即f(A)与f(B)相互之间没有继承关系。

  协变和逆变的概念实际上就只有这些,下面我们以具体的例子来看看协变和逆变的应用。

数组是协变的

  我们创建类型为 NumberInteger 类型的两个数组,并分别输出数组的类名,代码以及运行结果如下:

Number[] n1 = new Number[1];
Integer[] n2 = new Integer[1];
System.out.println(n1.getClass().getName());
System.out.println(n2.getClass().getName());
java.lang.Number;
java.lang.Integer;

  可以发现,其结果是协变的。

泛型是不变的(类型擦除)

  给定一个以泛型编程构建的列表List<T>,如果B是A的子类型,我们会习惯性的认为List<B>是List<A>的的子类型,List<A>中随意存放B类型的元素,但实际上由于泛型结构的特殊性,泛型只存在编译阶段,在运行时会进行类型擦除,导致泛型的不变性,List<A>和List<B>实际上是一种兄弟关

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值