Java 逆变与协变
最近一直忙于学习模电、数电,搞得头晕脑胀,难得今天晚上挤出一些时间来分析一下Java中的逆变、协变。Java早于C#引入逆变、协变,两者在与C#稍有不同,Java中的逆变、协变引入早于C#,故在形式没有C#直观(Google推出的基于jvm的Kotlin语音,则完全走向了C#的路线)。Java中逆变、协变,在泛型集合使用中更多些、更直观(像C#中的用法在Java中较少出现,但并非不可)。
正常泛型集合的使用
示例代码如下:
public static void main(String[] args) {
ArrayList<Number> alist = new ArrayList<>();
alist.add(new Integer(10));
alist.add(new Float(10.12));
Number n = alist.get(0);
}
对于alist集合而言,泛型类型参数 Number 已限定其可容纳的类型。Integer/Float 类型的对象都可成功加入,并能通过get(index)获取(获取时注意类型为Number)。通过这个示例,可以看到正常泛型集合可 "存取规范的类型及其子类型对象"。
协变
如下示例代码:
public class Person {
}
public class Student extends Person{
}
public class Teacher extends Person{
}
public static void main(String[] args) {
ArrayList<Person> alist = new ArrayList<>();
ArrayList<Student> slist = new ArrayList<>();
//类型转换错误
alist = slist;
}
在 Java 中 Person虽是Student的父类,但泛型 ArrayList<Person> 却并非 ArrayList<Student>的父类,所以 alist = slist 类型不兼容。上面程序中的目的是用 slist 替代 alist的实际功能,因此在以后使用 alist 时实际调用的应该是 slist 对应的功能。
在 Java 中要满足上述需求,应该具备2个条件:
1、放入数据时调用 alist.add(e) 要保证参数e,能转换为 slist.add(e) 中所需类型
2、获取数据时调用 alist.get(index) 的实际执行者 slist.get(index) 返回的结果能够转换为 alist 声明的Person类型
上面条件2是成立的,但条件1却并不一定成立。因 ArrayList<Person> 规范类型为Person,Student、Teacher都满足alist.add(e)的要求,却并不满足 slist.add(e)对类型必须是Student的要求。
修改代码如下:
public static void main(String[] args) {
ArrayList<? extends Person> alist = new ArrayList<>();
ArrayList<Student> slist = new ArrayList<>();
//类型转换正常
alist = slist;
//添加数据错误
alist.add(new Student());
//正常
Person p = alist.get(0);
}
? extends Person 指明 alist 赋值的泛型集合约束类型只要是 extends Person 的就可以,因此将 ArrayList<Student> slist 赋值给 alist 满足泛型类型参数约束。
alist.add(e) 根据 ? extend Person 的定义,e只要是 Person 类型或其子类型就可满足添加条件,然而 slist 只允许 Student 类型对象加入,因此前面所说的条件1无法保证(null可以被加入,但却意义) 。 因此Java中(根据前面2个条件分析可得到该结论)使用 extends 声明的泛型或泛型集合,只能从其中取值,不能向其中添值。
extends 声明的集合只能用于从其中取出元素,可能有朋友会问“不能向其中放入值,又何来从其中取值”。看下面的代码:
public static void main(String[] args) {
ArrayList<Student> slist = new ArrayList<>();
slist.add(new Student());
ArrayList<Teacher> tlist = new ArrayList<>();
tlist.add(new Teacher());
test(slist);
test(tlist);
}
private static void test(List<? extends Person> list) {
for(Person p : list){
System.out.println(p.toString());
}
}
通过上面的代码很容易发现,test方法的参数 list 在方法内部不允许添加元素(null除外);但在 main 中却可以向 slist、tlist 中添加元素,并作为实参传递到 test 方法的调用过程中。从 test 方法的角度分析,参数 list 里面具体存放的是Person?Student?Teacher?无所谓,这里统一当作Person对象来用绝对是类型安全的。这种把 泛型约束的子类型 当作 泛型约束的父类型 来用的情况就是协变。
逆变
示例代码:
public static void main(String[] args) {
ArrayList<Person> alist = new ArrayList<>();
ArrayList<Student> slist = new ArrayList<>();
//类型转换错误
slist = alist;
}
如上代码 意欲使用 alist 替代 slist 进行元素的存储,跟进前面的分析上面代码编译错误。
在 Java 中要满足上述需求,应该具备2个条件:
1、放入数据时调用 slist.add(e) 要保证参数e,能转换为 alist.add(e) 中所需类型
2、获取数据时调用 slist.get(index) 的实际执行者 alist.get(index) 返回的结果能够转换为 slist 声明的Student类型
上面条件1是成立的,但条件2却并不一定成立。因 ArrayList<Person> 规范类型为Person,Student、Teacher都满足alist.add(e)的要求,alist.get(index) 返回值却并一定满足 slist.get(index)返回值必须是Student类型的要求。
修改代码如下:
public static void main(String[] args) {
ArrayList<Object> alist = new ArrayList<>();
ArrayList<? super Person> slist = new ArrayList<>();
// 正常
slist = alist;
slist.add(new Student());
slist.add(new Teacher());
// 错误
slist.add(new Object());
Object obj = slist.get(0);
}
? super Person 指明 slist 赋值的泛型集合约束类型只要是 super Person 的就可以,因此将 ArrayList<Object> alist 赋值给 slist 满足泛型类型参数约束。 ? super Person 虽指明集合可容纳Person的父类类型(仅是有容纳能力),但 Person 的继承关系、层级并不明确(Person来自其他jar包)。因此该泛型约束对于Person的父类型并无约束能力,所以 super 禁止添加Person父类型到集合中(有能力容纳,但不允许放入),只能放入Person及其子类型。
slist.get(index)调用实际上执行的是alist.get(index),最终返回值类型是Person?Teacher?Student? 不得而知,所以使用 super 声明的泛型集合get返回只能为Object类型。
因此Java中(根据前面2个条件分析可得到该结论)使用 super 声明的泛型或泛型集合,只能向其中添值,不要从其中取值(取回的值均为 Object 类型,没有意义)。 同样有朋友会问 super 声明的泛型集合只放不取有何意义?看下面代码:
public static void main(String[] args) {
ArrayList<Person> alist = new ArrayList<>();
test2(alist);
for(Person p : alist){
System.out.println(p.toString());
}
}
public static void test2(List<? super Person> list){
list.add(new Student());
list.add(new Teacher());
}
通过上面的代码很容易发现,在 main 中 alist 有明确的类型,并作为实参传递到 test2 方法的调用过程中。从 test2 方法的角度分析,可以向集合中添加Person及其子类对象。这种把 泛型约束的父类型 当作 泛型约束的子类型 来用的情况就是逆变。
说明:
Java 中使用逆变、协变的机会并不算多。如果要使用请记住:如果限定集合仅可放入值时用 super、如果要限定集合仅可取出值时用 extends、如果集合既需要放入值又要取出值时用标准泛型集合。
文章写的仓促,若有不妥请各位朋友指正。
Java 逆变与协变的更多相关文章
- Java 逆变与协变的名词说明
最近在研究Thinking in Java的时候,感觉逆变与协变有点绕,特意整理一下,方便后人.我参考于Java中的逆变与协变,但是该作者整理的稍微有点过于概念化,我在这里简单的说一下 我对于协变于逆 ...
- Java中的逆变与协变
看下面一段代码 Number num = new Integer(1); ArrayList<Number> list = new ArrayList<Integer>(); ...
- Java中的逆变与协变(转)
看下面一段代码 Number num = new Integer(1); ArrayList<Number> list = new ArrayList<Integer>(); ...
- Java中的逆变与协变 很直接不饶弯的讲出来了
```java 协变 extends只能new 辈分比自己低的家伙 List<? extends Number> list001 = new ArrayList<Integer> ...
- Java中的逆变与协变 专题
结论先行: PECS总结: 要从泛型类取数据时,用extends: 协变 要往泛型类写数据时,用super: 逆变 既要取又要写,就不用通配符(即extends与super都不用) 不变 List&l ...
- Java逆变(Covariant)和协变(Contravariant)
1. 定义 逆变和协变描述的经过类型变换后的类型之间的关系.假如A和B表示类型,f表示类型变换,A ≤B表示A是B的子类型,那么 如果A ≤B,f(A) ≤f(B),那么f是协变 如果A ≤B,f(B ...
- C# 逆变与协变
该文章中使用了较多的 委托delegate和Lambda表达式,如果你并不熟悉这些,请查看我的文章<委托与匿名委托>.<匿名委托与Lambda表达式>以便帮你建立完整的知识体系 ...
- scala 学习: 逆变和协变
scala 逆变和协变的概念网上有很多解释, 总结一句话就是 参数是逆变的或者不变的,返回值是协变的或者不变的. 但是为什么是这样的? 协变: 当s 是A的子类, 那么func(s) 是func(A) ...
- Scala 深入浅出实战经典 第81讲:Scala中List的构造是的类型约束逆变、协变、下界详解
王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-97讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...
随机推荐
- ABP从入门到精通(2):aspnet-zero-core 使用MySql数据库
关于 asp.net zero core 项目的启动及说明,请观看我前面的博文 http://www.cnblogs.com/stulzq/p/7237153.html 本操作对于ABP默认项目应该也 ...
- Objectiv-C UIKit基础 NSLayoutConstraint的使用(VFL实现)
利用VFL可视化语言 (简单的抛砖引玉) 构建3个View 橙色和绿色左中右间隔20 上间隔40 高为200 蓝色在橙色内(0,0)处 宽高为橙色的一半 实现效果如下 由于atutosize和auto ...
- Opencv-2017-7-18
橘子薄皮只吃瓤,可以称之为过滤,意思是只要我们需要的东西,去除不需要的. 图像灰度级的分布及变化. 空间域(分布)和频域(变化). 低频(变化小),高频,水平/垂直,(高/低通滤波器). 低频-类似模 ...
- [IB]PeopleSoft异步详细信息中状态“已完成”但订阅合同状态“新建”问题
最近遇到一个IB异步程序状态不一致问题,异步详细信息中上面的状态是“DONE”但是订阅合同中还是“新建”状态.在域状态中清除域状态也不管用. 重启app server也不好使.最后执行了appmsgp ...
- node调用phantomjs-node爬取复杂页面
什么是phantomjs phantomjs官网是这么说的,'整站测试,屏幕捕获,自动翻页,网络监控',目前比较流行用来爬取复杂的,难以通过api或正则匹配的页面,比如页面是通过异步加载.phanto ...
- Vue2源码分析-逻辑梳理
很久之前就看完vue1,但是太懒就一直没写博客,这次打算抽下懒筋先把自己看过了记录下来,否则等全部看完,估计又没下文了 看源码总需要抱着一个目的,否则就很难坚持下去,我并没做过vue的项目,我几乎很少 ...
- Android文件上传与下载
文件上传与下载 文件上传 -- 服务端 以Tomcat为服务器,Android客服端访问Servlet,经Servlet处理逻辑,最终将文件上传,这里就是简单模拟该功能,就将文件上传到本机的D:\\u ...
- Python爬虫初学(二)—— 爬百度贴吧
Python爬虫初学(二)-- 爬百度贴吧 昨天初步接触了爬虫,实现了爬取网络段子并逐条阅读等功能,详见Python爬虫初学(一). 今天准备对百度贴吧下手了,嘿嘿.依然是跟着这个博客学习的,这次仿照 ...
- (3)markdown软件的使用
运行Mou.zip解压出来一个软件,它让托到应用程序中,然后打开 另一种软件为gitBook 安装好软件后,使用快捷键F4可以调出所有的应用程序 使用md(markdown简称)有个缺点就是,当内容比 ...
- 发布内网网站服务器让公网可以访问,无需NAT
有些时候,我们的测试网站搭建在我们的测试环境中,网站正式上线前,需要先测试下我们的测试网站是否正常,就可以用下面的方式将其内网网站服务器放至公网上,用器提供的外网地址就可以直接访问我们的内网网站服务器 ...