kotlin中的高级特性--协变与逆变(反变)

逆变性与协变性是kotlin中相对于java的新特性,这个成为不少java转kotlin学习的一个坎,在这篇文章里我将详细介绍和推导逆变性与协变性的由来。

内容参考了以下两篇博客:
http://www.cnblogs.com/lemontea/archive/2013/02/17/2915065.html
http://www.cnblogs.com/Ninputer/archive/2008/11/22/generic_covariant.html

在此之前我们需要明白一个大前提:

java不允许向下转型(父类转换成子类)

定义

假设有这样两个类型:TSub是TParent的子类,显然TSub型引用是可以安全转换为TParent型引用的。如果一个泛型接口IFoo<T>,IFoo<TSub>可以转换为IFoo<TParent>的话,我们称这个过程为协变,而且说这个泛型接口支持对T的协变。而如果一个泛型接口IBar<T>,IBar<TParent>可以转换为T<TSub>的话,我们称这个过程为反变(contravariant),而且说这个接口支持对T的反变。因此很好理解,如果一个可变性和子类到父类转换的方向一样,就称作协变;而如果和子类到父类的转换方向相反,就叫反变性。

我们来具体看一下体现到kotlin语法中是什么样的

kotlin中有out和in关键字来表示协变和逆变,我们通过out的两个来认识什么是逆变:
1. 泛型只能在返回值中出现
2. 只能进行子类向父类的转型

eg:
//有如下两个类
//1.不支持逆变与协变
MyFuncA<T>
//2.支持协变
MyFuncB<out T>
//现对其进行初始化然后转型
MyFuncA<object> funcAObject = null;
MyFuncA<string> funcAString = null;
MyFuncB<object> funcBObject = null;
MyFuncB<string> funcBString = null; 
funcAObject = funcAString;//编译失败,MyFuncA不支持逆变与协变
funcBObject = funcBString;//变了,协变
funcBObject = funcBInt;//编译失败,值类型不参与协变或逆变

代码中可以看出使用了协变的泛型对象MyFuncB<out T>可以进行子类向父类的转换,而不支持逆变和协变得MyFuncA<T>则不允许向上或者是向下的转换。

其实以上的两条含义只是一条,只不过在不同的场景下表现不一样而已,我们一起来看一下:

假设有这样一个方法:

String Base<out T>
{
  void Test(T t)
}

泛型协变的,但我们允许有方法可以在参数中使用泛型(****实际上这样是不行的,这里我们通过反正法证明来证明这一结论****)

Base<object> BaseObject = null;
Base<string> BaseString = null;
BaseObject = BaseString;
BaseObject.Test("");

我们来看一下函数的调用过程:

clipboard.png

****BaseObject****被****BaseString****初始化,所以
****BaseObject.Test("")****的调用实质上是调用****BaseString.Test("")****,而****BaseString****中要的泛型****T****是****string****,而实际****BaseObject****给出的泛型****T****是****object****,
****object****无法向下转型为****string****,因此出现类型转换的异常。

由此我们得出以上结论,因为泛型是协变的,进行子类向父类的转型,所以泛型不能在传入参数中使用,只能在返回值中使用。

逆变性反之也是一样的推导,由于进行的是父类向子类的转型,在返回值返回的时候要求的是子类的泛型,但实际上是调用父类的方法返回了父类,同样出现了向下转型的错误,因此逆变性中泛型只能在传入参数中使用,不能在返回值中使用。

eg:

//过程同上
T Base<in T>.Test()

****泛型逆变的,但我们允许有方法可以在返回值中使用泛型****(实际上这样是不行的,这里我们同样通过反正法证明来证明这一结论)

Base<object> BaseObject = null;
Base<string> BaseString = null;
BaseString = BaseObject ;
BaseString.Test();

只要按照协变时的调用方法看代码的调用就会发现我们在返回值的时候得到的是****object****,而我们要的是****string****,同样出现向下转型的错误。

逆变和协变是保证运行时安全而出现的机制,编码时编译器已经强制我们在逆变中不能在函数返回值中使用泛型,在协变中不能在函数参数中使用泛型,以保证运行时的安全,也就是将我们可能产生的类型转换异常在编译阶段给解决了!
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容