一,背景
常规情况下,使用泛型方法时不能直接访问泛型的具体类型,
举一个例子:定义一个泛型方法,这个方法的功能是打印 value 的值,同时打印value的 class 类型。
kotlin
fun <T> getValue(value: T) {
print("value 的值为 ${value};value 的类型为 ${T::class.java}")
}
这时会报错:Cannot use 'T' as reified type parameter. Use a class instead,意思是不能使用 T 作为具体化的类型。
如果我们想在方法内部访问泛型的具体类型怎么办?kotlin 中提供了关键字 reified,使用 reified 修饰泛型后,就允许我们在方法内部使用泛型的具体类型了。(注意reified需要搭配inline使用)
二,reified 的作用:在方法内部访问泛型的具体类型
如果想在方法内部访问泛型的具体类型,使用关键字 reified 修饰泛型,同时方法声明为内联函数,这样就可以在方法内部访问泛型的具体类型了,如下:
kotlin
inline fun <reified T> getValue(value: T) {
print("value 的值为 ${value};value 的类型为 ${T::class.java}")
}
然后调用该方法打印一个具体字符串:
kotlin
class Utils {
fun main() {
getValue("name")
}
inline fun <reified T> getValue(value: T) {
print("value 的值为 ${value};value 的类型为 ${T::class.java}")
}
}
打印输出:value 的值为name; value 的类型为String.class。
三,为什么在 reified 修饰的泛型 T 的方法内部可以拿到泛型的具体类型
上面的代码,反编译的代码如下:
ini
public final class Utils {
public final void main() {
Object value$iv = "name";
int $i$f$getValue = false;
String var4 = "value 的值为 " + value$iv + ";value 的类型为 " + String.class;
System.out.print(var4);
}
// $FF: synthetic method
public final void getValue(Object value) {
int $i$f$getValue = 0;
StringBuilder var10000 = (new StringBuilder()).append("value 的值为 ").append(value).append(";value 的类型为 ");
Intrinsics.reifiedOperationMarker(4, "T");
String var3 = var10000.append(Object.class).toString();
System.out.print(var3);
}
}
main 方法中直接内联了 getValue 方法的内部逻辑,直接将泛型的具体类型 String.class 硬编码到了 main 方法中,即访问到了调用泛型方法时泛型的具体类型。
四,使用场景:
1,封装 startActivity 方法
不使用 reified 时,启动 activity 需要 class 对象,我们必须手动传进来,因为我们无法知道泛型在使用时的具体类型。
kotlin
fun <T: Activity> Activity.startActivity(clazz: Class<T>) {
startActivity(Intent(this, clazz))
}
使用 reified 时,启动 activity 时可以直接拿到泛型方法使用时泛型的具体类型,不需要将 class 手动传进来。
kotlin
inline fun <reified T : Activity> Activity.startActivity() {
startActivity(Intent(this, T::class.java))
}