Kotlin中的reified关键字

前言

Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,被称之为 Android 世界的Swift,在Google I/O 2017中,Google 宣布 Kotlin 成为 Android 官方开发语言

img.jpg

reified关键字的作用

reified意为具体化,是kotlin所增强的一种泛型的使用方式,需要配合inline关键字使用

Java中的泛型

Java中我们使用泛型类型有泛型类,泛型接口,泛型方法,主要的目的就是去封装一个模板,让模板满足多种类型情况的调用,比如我们常用的集合类,不同的集合可以有不同的元素类型,比如:StringPersonNewsItem等,他们都会有相同的增 删 改 查等操作,唯一不同的就是元素类型不同,这种情况下就可以 将其类型抽离出来,将类型作为参数传入,这种就是典型的使用泛型类的方式

public class ArrayList<E>{}

再有例子就是我们定义了进行元素比较的接口,接口可能会用于不同类型的元素进行比较,则可以定义泛型接口,然后 在不同的实现类指定具体的泛型类型,这种就是典型的使用泛型接口的方式

public interface Compartor<T>{
    int compare(T o1, T o2);
}

一个函数也一样,如果我们一个函数会有不同的类型的参数调用,也可以将其定义为泛型方法

public <T extends View> T findViewById(@IdRes int id) {
    return getDelegate().findViewById(id);
}

泛型的类型擦除

Java中泛型的实现方式是通过编译器擦除泛型类型来实现的,即运行时不带任何泛型类型相关的信息,比如ArrayList<String>在运行时候只是用ArrayList<Object>来运行,只不过在读取的时候将Object转换为String

public void main() {
    ArrayList<String> list = new ArrayList<>();
    list.add("123");
    String value = list.get(0);
}

转化为字节码

 0 new #2 <java/util/ArrayList>
 3 dup
 4 invokespecial #3 <java/util/ArrayList.<init>>
 7 astore_1
 8 aload_1
 9 ldc #4 <123>
11 invokevirtual #5 <java/util/ArrayList.add>
14 pop
15 aload_1
16 iconst_0
17 invokevirtual #6 <java/util/ArrayList.get>
20 checkcast #7 <java/lang/String>
23 astore_2
24 return

在第20行将其转换成了String,也就是说其实不同的泛型类型执行的时候调用的都是一份代码,只不过在使用数据的时候多了一份转换,所以下面的写法是不允许的,所以是无法通过泛型类型拿到泛型类型的相关信息

public <T> void test(T t) {
    Class<T> c=T.class;//提示错误
    if (t instanceof String) {
        T t1 = new T() //提示错误
    }
}

使用reified关键字

  • 使用reified修饰函数中的泛型类型声明
  • 使用inline修饰函数使之内联

省略参数中的Class对象
如果你的泛型方法的参数含有Class对象,因为泛型函数中是无法获取到class对象的,所以需要

fun <T> test(t:T,clazz: Class<T>) {
    val c = T::class.java //报错
    // 因为无法从T拿到T的反射,需要传入参数clazz
}

使用reified关键字修饰

inline fun <reified T> test(t:T) {
    val c = T::class.java //编译成功
}

inline fun <reified T, V> test(t: T, v: V) {
    val c = T::class.java //编译成功
    val c1 = V::class.java //编译失败
}

inline修饰函数表示内联函数,会将泛型类型内嵌到编译代码中替换成具体类型,配合reified 关键字使用就可以在泛型函数中获取到反射类型的Class对象


inline fun <reified T : Activity> Activity.startActivity(context: Context) 
 = startActivity(Intent(context, T::class.java))

startActivity<LoginActivity>(context)

可以再看一个案例,比如我们需要封装在Fragment下的公共获取传参的参数,由于我们需要在方法内部判断获取数据的类型,所以需要将Class对象传入,因为默认是拿不到泛型的反射类型的

fun <T> Fragment.getValue(key: String, clazz: Class<T>) =
    when (clazz) {
        java.lang.String::class.java -> {
              arguments?.getString(key)
        }
        java.lang.Long::class.java -> {
            arguments?.getLong(key)
        }
        else -> {
            null
        }
    }

使用reified优化

inline fun <reified T> Fragment.getValue(key: String) =
    when (T::class) {
        String::class -> {
            arguments?.getString(key)
        }
        Long::class -> {
            arguments?.getLong(key)
        }
        else -> {
            null
        }
    }

或者我们在一些特殊的额情况下需要创建泛型类型的实例

fun <T> test()= T::class.java.newInstance() //编译错误,无法拿到Class

inline fun <reified T> test()= T::class.java.newInstance() //编译成功

Array中的使用
因为需要我们需要定义一个数组分割某个子元素的函数,以便方便使用,我们可以这样
会有编译错误提示Can't use T as a reified type parameter

fun <T> splitArray(array: Array<T>, item: T) =
   //这里会编译报错,提示Can't use T as a reified type parameter
    Array(array.size - if (array.contains(item)) 1 else 0) {
        if (it >= array.indexOf(item)) {
            array[it + 1]
        } else {
            array[it]
        }
    }

这种情况是因为,Array在编译时候会被编译成java T[],所以也需要将其进行实化

inline fun <reified T> splitArray(array: Array<T>, item: T) =
    Array(array.size - if (array.contains(item)) 1 else 0) {
        if (it >= array.indexOf(item)) {
            array[it + 1]
        } else {
            array[it]
        }
    }

欢迎关注Mike的简书

Android 知识整理

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,835评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,598评论 1 295
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,569评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,159评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,533评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,710评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,923评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,674评论 0 203
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,421评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,622评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,115评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,428评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,114评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,097评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,875评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,753评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,649评论 2 271

推荐阅读更多精彩内容