Kotlin基础 reified关键字

一,背景

常规情况下,使用泛型方法时不能直接访问泛型的具体类型,

举一个例子:定义一个泛型方法,这个方法的功能是打印 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))
}
相关推荐
用户091 天前
Gradle Cache Entries 深度探索
android·java·kotlin
循环不息优化不止1 天前
安卓 View 绘制机制深度解析
android
叽哥1 天前
Kotlin学习第 9 课:Kotlin 实战应用:从案例到项目
android·java·kotlin
雨白1 天前
Java 线程通信基础:interrupt、wait 和 notifyAll 详解
android·java
诺诺Okami2 天前
Android Framework-Launcher-UI和组件
android
潘潘潘2 天前
Android线程间通信机制Handler介绍
android
潘潘潘2 天前
Android动态链接库So的加载
android
潘潘潘2 天前
Android多线程机制简介
android
CYRUS_STUDIO2 天前
利用 Linux 信号机制(SIGTRAP)实现 Android 下的反调试
android·安全·逆向
CYRUS_STUDIO2 天前
Android 反调试攻防实战:多重检测手段解析与内核级绕过方案
android·操作系统·逆向