协变与逆变原理深度剖析及实例演示

在上一次https://www.cnblogs.com/webor2006/p/11279692.html中对于泛型中的协变与逆变进行了初步了解,由于这个概念是脱离开语言而且又非常之重要的,所以接下来继续对这个概念进行进一步巩固,达到一种通透领悟的境界。

这里还是从Java的角度再来看一下关于泛型协变与逆变的一个演变过程:

其错误提示为:

因为List<String>并不是List<Object>的子类型,为啥呢?这里可以采用假设法,假设这句赋值语句代码是成立的,那可以这样写:

而由于list2是指向了list,那么可以这样写:

很显然Object是不可能是String类型,在木有强制类型转换的情况下,所以说我们的这个假设是行不通的。

好,下面定义几个类来演示协变与逆变的情况,先定义三个类:一个父类,两个子类:

因为animals里面的元素接收的是Animal的子类型,当然满足要求。那接下来再来看一句代码:

看结果:

看一下错误提示:

为啥?还是采用假设法,假设此句代码是成立的,那会产生啥问题呢?由于animals里面的元素类型是Animal以及它的子类型,也就是有可能是Cat,也有可能是Dog,那我们在取元素时就会出现混乱,也就说明对于协变而言,只能对它进行读,不能进行写。

这里有一个术语:PECS:Producer Extends,Consumer Super。表示用于返回数据的生产者需要用Extends,而用于写入数据的消费者要用Super

下面来看一下逆变的情况:

逆变只能写,不能读,所以咱们可以往里写元素:

接下来试着取一下元素:

因为元素的类型是Animal及它以上的,所以从集合中拿的话很显然不一定就是Animal,那既然它里面是Animal之上的类型,是不是可以添加一个Object类型呢,试试:

不行的,Object因为不是Animal类型,所以没法往里写,既使层次体系满足Animal之上,如果改成这样写就成:

其实在Java中的数组是天然支持协变的,如何理解?下面看一下:

但是!!其实是会产生问题的,如下:

编译运行:

以上是关于Java层面关于泛型的协变和逆变的表现,下面则回到Kotlin来看下:

接下来来看下协变:

上面的代码其实相当于这两行代码:

好,接下来再来看一下逆变:

其实相当于Java的这段代码:

所以在Kotlin中:Consumer需要用in,而Producer需要用out。

posted on 2019-08-02 23:40  cexo  阅读(685)  评论(0编辑  收藏  举报

导航