Kotlin(1.1)学习笔记(6)——泛型

in和out

和java一样,kotlin中也有泛型的概念。不同的是,java中使用了通配符而kotlin中不存在。本文主要介绍两者的不同之处。
java中为了解决

List<String> strs = new ArrayList<String>();
List<Object> objs = strs; 

类似语句中strs复制objs报错的问题,使用了通配符,如常见的<? extends Object>,<? super Object>;在kotlin中则要在使用泛型出outin来限定泛型是否允许插入和读取。in修饰的泛型只能写入,out修饰的泛型只能读取。
举个例子:
我们定义一个copy方法,将第一个数组的内容拷贝到第二个数组中去

val ints: Array<Int> = arrayOf(1, 2, 3)
val any = Array<Any>(3)
copy(ints, any)
错误示例一:
fun copy(from: Array<Any>, to: Array<Any>) {
    assert(from.size == to.size)
    for (i in from.indices)
        to[i] = from[i]
}
这里调用copy(ints, any) // 错误:期望 (Array<Any>, Array<Any>),

因为Array<Int>不是Array<Any>的子类,编译器认为我们可能修改里面的信息,向其中加入非Any类型的对象,所以报错

正确示例一
fun copy(from: Array<out Any>, to: Array<Any>) {
 // ……
}

这个函数和上面那个唯一的不同就是参数from中增加了out关键字,它等同于java中的<? extends object>表明from这个对象是能用来做source,并读取里面的信息,不能向内增加数据

同样的,我们也可以为第二个参数增加修饰的in,等同于java中的<? super Object>,to只能接收Any以及其父类(ps:在本例中,虽然可以通过编译器,但是这样写已经无意义)
正确示例二

fun copy(from: Array<out Any>, to: Array<in Any>) {
 // ……
}

根据官网的介绍,我们可以把from称作生产者,to成为消费者。

星投影

看到这里我们已经学到了它们的基本用法,下面来将两个特殊的星投影

  • 对于 Foo <out T>,其中 T是一个具有上界 TUpper 的协变类型参数,Foo <*>等价于 Foo <out TUpper>。 这意味着当T 未知时,你可以安全地从 Foo <*>读取 TUpper 的值。
  • 对于 Foo <in T>,其中 T是一个逆变类型参数,Foo <*> 等价于Foo <in Nothing>。 这意味着当 T 未知时,没有什么可以以安全的方式写入Foo <*>
  • 对于 Foo <T>,其中 T 是一个具有上界 TUpper 的不型变类型参数,Foo<*>对于读取值时等价于 Foo<out TUpper>而对于写值时等价于 Foo<in Nothing>
    如果泛型类型具有多个类型参数,则每个类型参数都可以单独投影。 例如,如果类型被声明为interface Function <in T, out U>,我们可以想象以下星投影:
Function<*, String> 表示 Function<in Nothing, String>;
Function<Int, *> 表示 Function<Int, out Any?>;
Function<*, *> 表示 Function<in Nothing, out Any?>。

注意:星投影非常像 Java 的原始类型,但是安全。

(以下内容官网已经将的很详细,这里直接照抄)

泛型函数

不仅类可以有类型参数。函数也可以有。类型参数要放在函数名称之前:

fun <T> singletonList(item: T): List<T> {
    // ……
}

fun <T> T.basicToString() : String {  // 扩展函数
    // ……
}

要调用泛型函数,在调用处函数名之后指定类型参数即可:
val l = singletonList<Int>(1)

泛型约束

能够替换给定类型参数的所有可能类型的集合可以由泛型约束限制。
上界
最常见的约束类型是与 Java 的 extends 关键字对应的 上界:

fun <T : Comparable<T>> sort(list: List<T>) {
    // ……
}

冒号之后指定的类型是上界:只有 Comparable<T>
的子类型可以替代T
。 例如

sort(listOf(1, 2, 3)) // OK。Int 是 Comparable<Int> 的子类型
sort(listOf(HashMap<Int, String>())) // 错误:HashMap<Int,String> 不是 Comparable<HashMap<Int, String>> 的子类型

默认的上界(如果没有声明)是 Any?
。在尖括号中只能指定一个上界。 如果同一类型参数需要多个上界,我们需要一个单独的 where-子句:

fun <T> cloneWhenGreater(list: List<T>, threshold: T): List<T>
    where T : Comparable,
          T : Cloneable {
  return list.filter { it > threshold }.map { it.clone() }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 人生苦多,快来 Kotlin ,快速学习Kotlin! 什么是Kotlin? Kotlin 是种静态类型编程...
    任半生嚣狂阅读 26,303评论 9 118
  • 系列文章全部为本人的学习笔记,若有任何不妥之处,随时欢迎拍砖指正。如果你觉得我的文章对你有用,欢迎关注我,我们一起...
    我爱吃栗子啊阅读 6,775评论 5 14
  • 大家好,我是William李梓峰,欢迎加入我的Kotlin学习之旅。今天是我学习 Kotlin 的第十五天,内容是...
    William李梓峰阅读 1,928评论 -1 1
  • 今天前辈交给我了一个任务,把领导签好字的文件盖上戳,扫描出来。我就盖啊,扫啊,后来我师傅跟我说盖章要盖正,从盖章就...
    豆芽王子阅读 486评论 0 0
  • 镜中 只要想起一生中后悔的事 梅花便落了下来 比如看她游泳到河的另一岸 比如登上一株松木梯子 危险的事固然美丽 不...
    神经质的喵小姐阅读 1,080评论 0 1