一篇文章教你搞清楚——Kotlin-进阶---不变型,跟我一起手写EventBus吧

List定义如下:

// List带泛型的定义
public interface List extends Collection {
boolean add(E e);
E get(int index);
}

// List定义如下:
public interface List extends Collection {
boolean add(String e);
String get(int index);
}

// List定义如下:
public interface List extends Collection {
boolean add(CharSequence e);
CharSequence get(int index);
}

虽然String get(int index);返回值的范围比CharSequence get(int index);小,满足了第二个条件。

但boolean add(String e);接收参数的范围明显比boolean add(CharSequence e);要小,不满足第一个条件。

所以List中不是List的子类型。换句话说,把程序中的List都替换成List是不安全的,因为可能会存在这样的代码:

val spannable = Spannable()
val list: List = mutableListOf()
list.add(spannable)

如果把这里的List换成List,编译器就会报错。因为boolean add(String e);只会处理String类型的实参,Spannable超出了这个范围。

不变型


把上面的例子表达成更抽象的定义如下:

假设 泛型 类A 包含 类型参数T,即class A,而Type1是Type2的子类,如果A和A不存在父子关系,则称 类A 在类型参数上是不变型的。

Kotlin 和 Java 中的类都是不变型的。

这样会造成一些限制,辛辛苦苦抽象了一个方法,它接收一个List作为参数:handle(chars: List),想当然地把List传递进入时,编译器会报错。。。难道需要为List重新定义一个相同的方法吗?

协变


不变型描述的是泛型类之间没有子类型关系,泛型类之间还有一种子类型关系叫协变。

协变的意思是:类与其类型参数的抽象程度具有相同的变化方向。

(试图总结某个抽象概念时,总会说出一些让人听不懂话。。。)

换句话说:当类型参数变得更具体时,类也变得更具体。当类型参数变得更抽象时,类也变得更抽象。

比如,从List到List,类型参数从String变为更抽象的CharSequence,如果List到List的变化方向也是更抽象(前者是后者的子类),就称List在类型参数T上是协变的。(显然这个例子不是协变的而是不变型的)

如果一个泛型类是协变的,就意味着它在类的层面保留了类型参数的子类型关系

Kotlin 中,声明类在类型参数上是协变的,需要添加out保留字:

class MyList{ … }

虽然将泛型类声明为协变可以让其子类型化关系更符合直觉,但这需要付出代价:

class MyList {
fun set(item: T) {}//报错: Type parameter is declare as “out” but occur at “in” position in type T
fun get(): T {…}
}

  • 若T出现在方法的参数位,称set(item: T)消费类型为T的值。

  • 若T出现在返回值位时,称get(): T生产类型为T的值。

当T被out修饰后,它只能出现在返回值位,即它只能被泛型类生产而不能被消费。
所以out会产生两个效果:

1.它保留了泛型类的子类型化。
2.它限制了类型参数只能出现在返回值位。

这两点是相辅相成的:正因为它限制了类型参数不能出现在参数位,所以子类型化得以保留。正因为它保留了子类型化,所以类型参数只能出现在返回值位。

假设类型参数出现在了参数位,就会出现在这样的情况:

class MyList {
fun set(itme: String)
fun get(): String
}

class MyList {
fun set(itme: CharSequence)
fun get(): CharSequence
}
复制代码
因为fun set(itme: String)可以接收的参数范围比fun set(itme: CharSequence)小,不符合第一条退推论,所以MyList不是MyList的子类型。
而添加了out之后,相当于告诉编译器把出错的方法删掉以保留子类型化:
class MyList {
fun get(): T {…}
}

class MyList {
fun get(): String
}

class MyList {
fun get(): CharSequence
}

此时fun get(): String返回值的范围比fun get(): CharSequence小,符合第二条推论,所以MyList是MyList的子类型。

逆变


除了不变型、协变,泛型类之间还有一种子类型关系:逆变。

逆变的意思是:类与其类型参数的抽象程度具有相反的变化方向。

换句话说:当类型参数变得更具体时,类却变得更抽象。当类型参数变得更抽象时,类却变得更具体。
逆变有一点反直觉,它想实现的效果是:List成为List是的子类型。

如果一个泛型类是逆变的,就意味着它在类的层面反转了类型参数的子类型关系

Kotlin 中,声明类在类型参数上是逆变的,需要添加in保留字:

class MyList{ … }

同样地,这需要付出代价:

class MyList {
fun set(item: T) {}
fun get(): T {…}//报错: Type parameter is declare as “in” but occur at “out” position in type T
}

当T被in修饰后,它只能出现在参数位,即它只能被泛型类消费而不能被生产。
由此可见:

out和int不仅限定了参数可以出现的位置,还限定了什么类可以成为子类型。

类型投影


生活中的投影,是把一个三维物体变成二维物体,投影看上去还是那个物体只是降了一维。
程序中的类型投影也是类似的意思:

将类型投影意味着保留该类型的有些能力,去掉另一些能力。通过类型投影可以动态地改变泛型类的子类型关系。

类型投影通常应用于将不变型的泛型类动态地转换成逆变或协变。
比如,MutableList就是不变型的:

public interface MutableList : List, MutableCollection {
// 类型参数出现在 out 位置
public fun removeAt(index: Int): E
// 类型参数出现在 in 位置
public fun add(index: Int, element: E): Unit

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

最后

分享一份工作1到5年以上的Android程序员架构进阶学习路线体系,希望能对那些还在从事Android开发却还不知道如何去提升自己的,还处于迷茫的朋友!

  • 阿里P7级Android架构师技术脑图;查漏补缺,体系化深入学习提升

  • **全套体系化高级架构视频;**七大主流技术模块,视频+源码+笔记

有任何问题,欢迎广大网友一起来交流

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
img
mg-1qAPZTMY-1712800914995)]

有任何问题,欢迎广大网友一起来交流

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-lujXPqjO-1712800914995)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值