5分钟Java短文:泛型之逆变和协变

开发 后端
日常琐碎的时间下,不适合看一些长篇高质量的文章,但是琐碎时间也是时间,看一些短小精悍的文章来查缺补漏也是极好的。碎花化的时间,就交给“碎片化的文章”来填充吧。

[[344440]]

本文转载自微信公众号「咸鱼正翻身」,作者MDove  。转载本文请联系咸鱼正翻身公众号。 

前言

日常琐碎的时间下,不适合看一些长篇高质量的文章,但是琐碎时间也是时间,看一些短小精悍的文章来查缺补漏也是极好的。碎花化的时间,就交给“碎片化的文章”来填充吧。

今天“碎片化文章”主题:泛型-逆变和协变。逆变和协变摆在这我猜很多朋友会蒙蔽,毕竟我们日常好像、大概没怎么接触过这个概念。

事实并非如此,我们日常开发中经常见,只是不知道这么个名词而已。

正文

OK,今天5分钟短文就让咱们聊一聊逆变和协变这俩个概念。

1、基础概念

其实它们俩的概念很好理解。接下来让我们仔细读一遍下边的这一段话:

逆变与协变用来描述类型转换后的继承关系。如果A、B表示类型,f(...)表示类型转换,≤表示继承关系(比如,A≤B表示A是B的子类)

如果f(...)是逆变的,那么当A≤B时则f(B)≤f(A)成立

如果f(...)是协变的,那么当A≤B时则f(A)≤f(B)成立

额外补充一条:如果f(...)是不变的,那么当A≤B时则f(B)与f(A)没有任何关系

2、代码场景

如果大家充分理解了上边的话,其实就能想到咱们日常代码中的例子:数组就是一种协变;泛型是不变的。上代码:

public class A extends B {} 
 
public class B {} 
 
 
 
 
public void test() { 
 
B[] arrs = new A[66]; 
 
List<B> list = new ArrayList<A>(); 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.

这段代码是编不过的:

 

因为数组是协变的,所以 A[]是 B[]的子类;而泛型不是,所以 List并不是 List的子类。

3、通配符的意义

因为这个原因的存在,所以才有了通配符。

3.1、协变-上限通配符

代码改成这个样子就可以正常编译了:

 

通配符的存在,让泛型产生了协变,让 List可以变成 List的子类。不过我猜经验丰富的同学已经知道,这样搞“没什么卵用”,因为:

 

我们发现,这样搞完。对于 list变量来说,我们只能 get()不能 add()!一时接受不了?其实这里也很好理解,协变之后对于list来说,我可以指向很多 List的子类。

假设此时我们可以随意 add(),那么对于运行期来说简直是灾难:因为我可以随意的 add(newA());add(newC())。如果这种情况存在那么我 get()的时候,是不是只能把它当做 B来使用,因为这里有可能有 A也有可能有 C...

这样搞完全没有意义...因此也就有了下边的内容:逆变-下限通配符

3.2、逆变-下限通配符

直接上代码:

public class A extends B {} 
 
public class B {} 
 
public class C extends B {} 
 
public class D extends A {} 
 
 
 
 
public void test2(List<? super A> list){ 
 
list.add(new A()); 
 
list.add(new B()); 
 
list.add(new C()); 
 
list.add(new D()); 
 

  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.

此时我们会发现:我们可以 add(), A及其子类。而这种实现就脱胎于咱们逆变这个概念。

 

3.3、小思考

如果我们仔细想一想会发现,这些都是在开发阶段或者编译阶段的限制。做了这么多限制,到底为了什么?或者收益是什么样的呢?

关于这部分内容的讨论,咱们后续再聊~

尾声

到此想聊的内容就结束了,关于泛型的话题还有很多很多,而熟练的使用和理解泛型对咱们编写工具、框架有着关键的帮助。

 

责任编辑:武晓燕 来源: 咸鱼正翻身
相关推荐

2022-04-18 20:12:03

TypeScript静态类型JavaScrip

2021-07-01 06:47:30

Java泛型泛型擦除

2013-05-03 10:57:09

泛型泛型教程

2024-06-19 09:58:29

2009-08-03 18:24:28

C# 4.0协变和逆变

2009-05-27 11:30:20

C#Visual Stud协变

2012-06-28 10:26:51

Silverlight

2019-12-12 10:25:33

Java泛型编程语言

2020-03-17 07:41:50

ApacheKafka系统

2011-01-14 10:27:18

C#.netasp.net

2012-03-13 09:32:15

C#协变

2020-07-25 19:40:33

Java开发代码

2025-02-17 10:41:16

2021-01-29 11:43:53

SSHLinux命令

2020-09-14 11:30:26

HTTP3运维互联网

2021-04-30 16:23:58

WebRTC实时音频

2020-11-23 16:23:59

CSS设计技术

2020-12-17 10:00:16

Python协程线程

2021-03-12 09:45:00

Python关联规则算法

2021-01-29 11:25:57

Python爬山算法函数优化
点赞
收藏

51CTO技术栈公众号