kotlin高级篇

反射

  • 什么是反射
    kotlin的反射和java反射差不多,第一个作用是获取程序在运行时刻的内部结构,比如检查一个对象的类型,某个类型有哪些构造函数、方法、属性,函数的类型,已经对应的类型参数等。第二个作用是在运行时对一个Java对象进行操作,这些操作包括动态创建一个类型的对象,获取某个属性的值以及调用方法。
  • 作用
    很多有名的java框架都是通过反射来实现的,比如spring框架,以及我们熟悉的retrofit是通过动态代理的机制实现的,而动态代理底层是通过反射实现的。

在 Java 平台上,使用反射功能所需的运行时组件作为单独的 JAR 文件( kotlin- reflect.jar )分发。如果你需要使用反射,请确保该 .jar文件添加到项目的 classpath 中。

  • 类引用
    kotlin类的元类型是KClass,而java是Class。
 val c: KClass = MyClass::class

//获取对象的类型
val myClass = MyClass()
val c1 = myClass:class

还可以获取下面的信息:

c.isAbstract
c.isCompanion
c.isFinal
c.isOpen
c.supertypes //List<KType>
c.typeParameters //List<KTypeParameter> 返回类型参数
c.constructors //Collection<KFunction<T>>
c.primaryConstructor //KFunction<T>?
c.memberProperties //Collection<KProperty1<T, *>>
c.declaredFunctions //Collection<KFunction<*>>
...

注意 constructors返回的是KFunction<T>类型的对象。
也可以获取java类型:

val javaC: Class<MyClass> = MyClass::class.java
  • 函数引用
fun isOdd(x: Int) = x % 2 != 0
fun isOdd(s: String) = s == "brillig" || s == "slithy" || s == "tove"

val numbers = listOf(1, 2, 3) 
println(numbers.filter(::isOdd)) // 引用到 isOdd(x: Int)

你也可以将函数引用存储在变量里:

val predicate: (String) -> Boolean = ::isOdd

如果对函数类型不清楚的同学可以先了解 lambda表达式和匿名函数

示例:函数组合

//函数组合
//compose(f, g) = f(g(*))
fun <A, B, C> compose(f: (B) -> C, g: (A) -> B): (A) -> C {
    return { x -> f(g(x)) }
}

fun length(s: String) = s.length
val oddLength = compose(::isOdd, ::length)
val strings = listOf("a", "ab", "abc")

println(strings.filter(oddLength)) // 输出 "[a, abc]"
  • 属性引用
    要把属性作为 Kotlin中 的一等对象来访问,我们也可以使用 :: 运算符:
var x = 1
fun main(args: Array<String>) { 
    println(::x.get()) // 输出 "1" 
    ::x.set(2)
    println(x) // 输出 "2"
}

表达式 ::x 求值为 KProperty<Int> 类型的属性对象,它允许我们使用 get() 读取它的 值,或者使用 name 属性来获取属性名。
对于可变属性,例如 var y = 1::y 返回 KMutableProperty<Int> 类型的一个值, 该类型 有一个 set() 方法。
属性引用可以用在不需要参数的函数处:

val strs = listOf("a", "bc", "def") 
println(strs.map(String::length)) // 输出 [1, 2, 3]

当我看到官网文档说上面也可以,我就懵逼了,map函数的参数是一个 (T) -> R 的函数类型,为什么String::length 却可以正确输出呢?最后我发现其实 String::length 返回的是 KProperty1<String, Int> ,然后我看他的源码就恍然大悟:

public interface KProperty1<T, out R> : KProperty<R>, (T) -> R {
  ...  
  public fun get(receiver: T): R
  ...
}

上面的代码相当于:

val get: (String) -> Int = {it.length}
val strs = listOf("a", "bc", "def")
println(strs.map(get))

要访问属于类的成员的属性,我们这样限定它:

class A(val p: Int)

fun main(args: Array<String>) {
  val prop = A::p
    println(prop.get(A(1))) // 输出 "1" 
}

对于扩展属性:

val String.lastChar: Char
    get() = this[length - 1]

fun main(args: Array<String>) {
    println(String::lastChar.get("abc")) // 输出 "c"
}
  • 构造函数引用
    构造函数可以像方法和属性那样引用。通过使用 :: 操作符并添加类名来 引用构造函数:
class Foo
fun function(factory: () -> Foo) {
    val x: Foo = factory()
}
fun main(args:Array<String>) {
    function(::Foo)
}

其实上面 ::Foo 返回的是 () -> Foo 函数类型的值。

  • 绑定的函数与属性引用
    你可以引用特定对象的实例方法。
val numberRegex = "\\d+".toRegex() 
println(numberRegex.matches("29")) // 输出“true”
val isNumber = numberRegex::matches 
println(isNumber("29")) // 输出“true”

上面的代码val isNumber = numberRegex::matches 会将matches的引用存储在 isNumber变量中,这样的引用会绑定到其接收者上。它可 以直接调用(如上例所示)或者用于任何期待一个函数类型表达式的时候:

val strings = listOf("abc", "124", "a70") 
println(strings.filter(isNumber)) // 输出“[124]”

比较绑定的类型和相应的未绑定类型的引用。 绑定的引用有其接收者“附加”到其上, 因此接收者的类型不再是参数:

val isNumber: (CharSequence) -> Boolean = numberRegex::matches
val matches: (Regex, CharSequence) -> Boolean = Regex::matches

属性引用也可以绑定:

val prop = "abc"::length 
println(prop.get()) // 输出“3”

协程

点击查看

类型安全的构建器

...

注解

...

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

推荐阅读更多精彩内容