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))
}
相关推荐
xiangzhihong812 分钟前
使用Universal Links与Android App Links实现网页无缝跳转至应用
android·ios
车载应用猿1 小时前
基于Android14的CarService 启动流程分析
android
没有了遇见1 小时前
Android 渐变色实现总结
android
雨白4 小时前
Jetpack系列(四):精通WorkManager,让后台任务不再失控
android·android jetpack
mmoyula6 小时前
【RK3568 驱动开发:实现一个最基础的网络设备】
android·linux·驱动开发
sam.li7 小时前
WebView安全实现(一)
android·安全·webview
移动开发者1号7 小时前
Kotlin协程超时控制:深入理解withTimeout与withTimeoutOrNull
android·kotlin
程序员JerrySUN8 小时前
RK3588 Android SDK 实战全解析 —— 架构、原理与开发关键点
android·架构
移动开发者1号8 小时前
Java Phaser:分阶段任务控制的终极武器
android·kotlin
哲科软件17 小时前
跨平台开发的抉择:Flutter vs 原生安卓(Kotlin)的优劣对比与选型建议
android·flutter·kotlin