Android Kotlin的Class、反射、泛型

前言

最近在学习kotlin的反射的时候遇到了一些问题,特地记录一下。

正题

在Java中使用Class很常见的就是,xxx类.class,比如我们在startActivity的时候startActivity(new Intent(this, OtherActivity.class)); 这里接收的就是CLass<?> cls参数。
那么在java中获取Class的方法有哪些呢?

1、Class c = person.getClass(); //对象获取
2、Class cc =Person.class;//类获取

而我们来看看kotlin

//对象获取
person.javaClass// javaClass
person::class.java // javaClass
//类获取
Person::class// kClass
 person.javaClass.kotlin// kClass
(Person::class as Any).javaClass// javaClass
Person::class.java // javaClass

哇,这么多种,他们是不是一样的,有没有什么区别?
log看看他们到底是不是相同的Class

println(person.javaClass == person::class.java) //true
println(person.javaClass == Person::class.java)//true
println(person::class.java == Person::class.java)//true
 //person.javaClass == person::class.java == Person::class.java
println(person.javaClass == Person::class)//false
println(person.javaClass.kotlin == Person::class)//true
println(person::class == Person::class)//true

从log来看,person.javaClass == person::class.java == Person::class.java
三者是相同的。但是person.javaClass == Person::class却是不同的。为什么呢?

原因是在kotlin中的Class与Java不同,kotlin中有一个自己的Class叫做KClassperson::classPerson::class都是获取kotlin的KClass,所以println(person::class == Person::class) 为true。
我们可以从kotlin的KClass获取到java的Class,person::class.java就是如此,先获取到kotlin的KClass然后再获取javaClass。
object/class->kClass->Class
同样也可以通过java的Class获取kotlin的KClass,person.javaClass.kotlin就是先获取javaClass然后再获取kotlin的KClass
object/class->Class->KClass
那么KClass都有些什么用呢?Find Usages 可以看到


几乎多数跟Reflect相关,而用的最多的也是在KClasses里面
KClasses扩展了许多跟反射相关的方法,算的上是kotlin的反射中类主力输出。

如果要使用kotlin的反射类的话,要加入

  compile "org.jetbrains.kotlin:kotlin-reflect:$kotlin_version"
举个例子

我们要把一个类的所有字段通过反射给打印出来,调用java的方法来实现是这样

 println (Person::class.java.declaredFields.map {
            it.isAccessible = true
            "${it.name}: ${it.get(person)}"
        }.joinToString(","))

我们通过Person::class拿到KClass,然后.java拿到java的Class<?>,再获取declaredFields,最后通过map,然后把获取的Field获取到值打印出来

问题

而使用kotlin,我们还有别的做法

 println (Person::class.memberProperties.map {
            it.isAccessible = true
            "${it.name}: ${it.get(person)}"
        }.joinToString(","))

通过Person::class拿到KClass,直接调用KClass的memberProperties来拿到KProperty的Collection集合,然后进行操作。当然,这里KProperty也是kotlin的反射类中的,也类似于Java的Field。
那么既然可以这样,理论上这样也应该是可以的

//person是对象不是Person类
 println (person::class.memberProperties.map {
            it.isAccessible = true
            "${it.name}: ${it.get(person)}"
        }.joinToString(","))

这下却出错了,真的奇怪!

Error:(73, 31) Out-projected type 'KProperty1<out OtherActivity.Person, Any?>' prohibits the use of 'public abstract fun get(receiver: T): R defined in kotlin.reflect.KProperty1'

为什么会这样?
查看it发现

kotlin的property的Person为out逆变的,R只能作为输出,不能作为get的参数传入,所报错了。
这里的kotlin泛型还是有点小坑的需要你踩一踩,网上有一篇文章解释说明了一番

那么怎么改呢?
三种办法:

//扩展KProperty的get方法为getUnsafed,其实就是去掉了out
   fun <T, R> KProperty1<T, R>.getUnsafed(receiver: Any): R {
        return get(receiver as T)
    }
//然后
println(person::class.memberProperties.map {
            it.isAccessible = true
            "${it.name}: ${it.getUnsafed(person)}"
        }.joinToString(","))
//强转一下
 println(person::class.memberProperties.map {
            it.isAccessible = true
            it as KProperty1<Person, Any>
            "${it.name}: ${it.get(person)}"
        }.joinToString(","))
//这一种就涉及到kotlin中的获取KClass的方式, 先获取java的Class再获取kotlin的KClass
//神奇的是,这种获取到的it的类型没有out,而是我们期望的 KProperty1<Person, Any>
 println(person.javaClass.kotlin.memberProperties.map {
            it.isAccessible = true
            "${it.name}: ${it.get(person)}"
        }.joinToString(","))

最后

在kotlin学习和探索的过程中发现了这些问题,非常地好奇,想要探知其原理。所以摸索着前进,希望能带给其他人一些学习的思路和兴趣。
另外,还有一个问题就是:
为什么有时候map的it是带out的有时候不带?这个跟class是java的class还是kotlin的class是否有关系?其间的原理和过程是怎么样的,我还在继续探索,希望明白的同学可以分享一下。

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

推荐阅读更多精彩内容

  • 前言 人生苦多,快来 Kotlin ,快速学习Kotlin! 什么是Kotlin? Kotlin 是种静态类型编程...
    任半生嚣狂阅读 26,139评论 9 118
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,431评论 25 707
  • 第九章 Kotlin与Java混合调用 虽然 Kotlin 的开发很方便,但当你与他人协作时,总会碰到 Java ...
    光剑书架上的书阅读 1,606评论 2 11
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,594评论 18 139
  • 你在公园里看过日出吗? 若这样提问,多数人会显出不屑,他们会说,看日出,最好的去处当是泰山,当是黄山,当是海上,当...
    雪琴吟阅读 1,118评论 2 9