在Kotlin中,总的来说,可以理解泛型为:
①向上解决类型不通用
②向下解决类型限定
如果看着累,建议先看小结,寥寥几字,大致了解下。
需要理解几个问题?
① 理解 型变是什么?逆变又是什么?
② 泛型存在的价值是什么?
③ 泛型注解out、in有什么用?Invariant又是什么?
④ 类型擦除 有什么影响?
本节的目录结构是这样的
Kotlin的泛型,功能与 Java 一样
fun main(args: Array) {
val age = 23
val name = "runoob"
val bool = true
doPrintln(age) // 整型
doPrintln(name) // 字符串
doPrintln(bool) // 布尔型
}
fun doPrintln(content: T) {
when (content) {
is Int -> println("整型数字为 $content")
is String -> println("字符串转换为大写:${content.toUpperCase()}")
else -> println("T 不是整型,也不是字符串")
}
}
输出了什么呢
整型数字为 23
字符串转换为大写:RUNOOB
T 不是整型,也不是字符串
这便是泛型了,范例中允许接受不同的类型,类型通用。
即类型转变。通过转变类型,提升API的适配度。
我们来看看Java的型变是怎么样的。
即在声明变量、参数时就确定T的类型。
// Java
void demo(Source strs) {
Source
// Kotlin
interface Source {
fun nextT(): T
}
fun demo(strs: Source) {
val objects: Source = strs // 这个没问题,因为 T 是一个 out-参数
// ……
}
注意看这里Source部分
在Source是使用处,Kotlin的型变解决了 子类型化的麻烦。
这里引入一个概念,型变注解:这里的out
同样的型变注解还有 in
投影就是 某事物的影子,影子之下才有天地。
只有在影子内的东西方可以动。
说的就是 限定了。
这讲的是一种将 某个类型投影 到某处。
你想想一下,只允许投影之下的暗处,这不正是限制吗?
类型投影,不就是类型限制吗?怎么限定呢?通过out\in的类型注解。
fun copy(from: Array, to: Array) {
assert(from.size == to.size)
for (i in from.indices)
to[i] = from[i]
}
fun main() {
val ints: Array = arrayOf(1, 2, 3)
val any = Array(3) { "" }
copy(ints, any) // 这里编译是不通过的,ints类型为 Array 但此处期望 Array
}
Array 在 T 上是不型变的,因此 Array 和 Array 都不是另一个的子类型。
为什么? 再次重复,因为 copy 可能做坏事。为了避免这种错误,编译器禁止了这种操作。
我们唯一要确保的是 copy() 不会做任何坏事。我们想阻止它写到 from去。
fun copy(from: Array, to: Array) { …… }
这便是类型投影。所有out(协变)是输出,in(逆变)是输入。
对于 out 泛型,可以将使用子类泛型的对象 -> 赋值给使用父类泛型的对象。
对于 in 泛型,可以将使用父类泛型的对象 -> 赋值给使用子类泛型的对象。
这使用于并不知道类型参数的任何信息。 * 代表通吃。
这个泛型类型的所有的实体实例, 都是这个投射的子类型。
Function<*, String>
这里用到了*
var list:ArrayList<> = arrayListOf(1) //<>必不可少 相当于java的无泛型
当 cList:ArrayList<> = javaList 时<>相当于
当 javaList:ArrayList<> = cList 时<>相当于
这就是什么都接收了
在简单的学习下,泛型的函数是怎么编写的
fun singletonList(item: T): List {
// ……
}
fun T.basicToString(): String { // 扩展函数
// ……
}
这么用
val l = singletonList(1)
可以省略能够从上下文中推断出来的类型参数
val l = singletonList(1)
这跟java是类似的
//泛型约束 <占位符:类型>
fun play(vararg param: T):Double{
return param.sumByDouble { it.toDouble() }
}
//多个约束,T有多个上限 , where T:类型,T:类型
fun getBetterBig(list:Array,threhold:T):List where T:Number,T:Comparable{
return list.filter { it>= threhold }.sorted()
}
泛型声明的类型安全检测是仅在编译期进行的。
例如,Foo 与 Foo
运行时泛型类型的实例不保留关于其类型实参的任何信息。
运行时不保留其类型实参的任何信息,即类型擦除。
编译器会禁止由于类型擦除而无法执行的 is 检测
fun handleStrings(list: List) {
if (list is ArrayList) {
// `list` 会智能转换为 `ArrayList`
}
}
泛型函数调用的类型参数也同样只在编译期检
这便是
在编译器会进行安全类型检查,在运行期则擦除类型,相当于是*类型
泛型,牵扯到 Out (协变)、In(逆变)、Invariant(不变),这是必须要理解的部分。
所谓泛型,即 类型限制,类型转变的称呼。