Kotlin泛型的高级特性

Kotlin中是可以将内联函数中的泛型进行实化的。
那么具体该怎么写才能将泛型实化呢?首先,该函数必须是内联函数才行,也就是要用inline
关键字来修饰该函数。其次,在声明泛型的地方必须加上reified关键字来表示该泛型要进行
实化。

inline fun <reified T> getGenericType() {
}

getGenericType()函数直接返回了当前指定泛型的实际类型。T.class这样的语法在Java
中是不合法的,而在Kotlin中,借助泛型实化功能就可以使用T::class.java这样的语法了。

inline fun <reified T> getGenericType() = T::class.java
fun main() {
val result1 = getGenericType<String>()
val result2 = getGenericType<Int>()
println("result1 is $result1")
println("result2 is $result2")
}

运行结果:
result1 is class java.lang.String
result2 is class java.lang.Integer

Android中的实际用法:

inline fun <reified T> startActivity(context: Context) {
	val intent = Intent(context, T::class.java)
	context.startActivity(intent)
}

startActivity<TestActivity>(context)

泛型的协变

参数列表是接收数据的地方,因此可以称它为in位置
返回值是输出数据的地方,因此可以称它为out位置
image
定义了一个MyClass<T>的泛型类,其中A是B的子类型,同时 MyClass<A> 又是 MyClass<B> 的子类型,那么我们就可以称MyClass在T这个泛型上是协变的。

在泛型T的声明前面加上了一个out关键字。这就意味着现在T只能出现在out位置上,而不能出现在in位置上,同时也意味着SimpleData在泛型T上是协变的。

open class Person(val name: String, val age: Int)
class Student(name: String, age: Int) : Person(name, age)
class Teacher(name: String, age: Int) : Person(name, age)

class SimpleData<out T>(val data: T?) {
fun get(): T? {
	return data
}
}

fun main() {
	val student = Student("Tom", 19)
	val data = SimpleData<Student>(student)
	handleMyData(data)
	val studentData = data.get()
	println(studentData?.name)
	println(studentData?.age)
	}
	fun handleMyData(data: SimpleData<Person>) {
		val personData = data.get()
	}

泛型的逆变

泛型在协变时只出现在out位置上,逆变时只出现在in位置上,就不会存在类型转换异常的情况

interface Comparable<in T> {
operator fun compareTo(other: T): Int
}
posted @ 2021-05-23 22:30  西北野狼  阅读(369)  评论(0编辑  收藏  举报