通过反射调用函数需要 KFunction
实例,KFunction
实例可以通过两种方式获得:一种是函数引用;另外一种是通过 KClass
提供的 API 获得 KFunction
实例。
函数引用可以表示一个函数字面量,可以赋值给函数类型变量。函数引用也是使用 ::
运算符,可以引用顶层函数也可以引用类中的成员函数。
- 通过反射调用函数的示例代码如下:
class Person() {
var name: String? = null
var age: Int = 0
fun setNameAndAge(name: String, age: Int) {
this.name = name
this.age = age
}
override fun toString(): String {
return "Person [name=$name, age=$age]"
}
}
fun add(a: Int, b: Int): Int = a + b
fun sub(a: Int, b: Int) = a - b
fun calculate(opr: Char): (Int, Int) -> Int = if (opr === '+') ::add else ::sub
fun main(args: Array<String>) {
val fn1 = ::add // 1️⃣
val fn2 = ::sub // 2️⃣
val fn3 = calculate('+') // 3️⃣
println(fn1.call(10, 5)) // 4️⃣15
println(fn1(10, 5)) // 5️⃣15
println(fn2(10, 5)) // 5
println(fn3(10, 5)) // 15
val clz = Person::class // 6️⃣
val person = clz.createInstance() // 7️⃣
clz.functions.forEach { println(it.name) } // 8️⃣
val pfn1 = clz.functions.first()
pfn1.call(person, "小三", 18)
println(person)
val pfn2 = Person::setNameAndAge
pfn2.call(person, "小五", 15)
println(person)
pfn2.call(person, "小雪", 20)
println(person)
}
运行结果:
15
15
5
15
setNameAndAge
toString
equals
hashCode
Person [name=小三, age=18]
Person [name=小五, age=15]
Person [name=小雪, age=20]
Process finished with exit code 0
上面代码第1️⃣、2️⃣行都是引用顶层函数。代码第3️⃣行调用 calculate
函数,该函数的返回类型是函数类型 (Int, Int) -> Int
,函数引用可以作为函数字面量表示函数,所以 calculate
函数直接返回函数引用。这也说明了 KFunction
类型与函数类型是兼容的。
代码第4️⃣行是通过 KFunction
的 call
函数调用引用的函数,也可以通过第5️⃣行直接调用函数。
代码第6️⃣行是引用 Person
类实例,代码第7️⃣行是通过引用类创建实例 Person
,createInstance
函数能够创建引用类的对象。代码第8️⃣行 functions
是 KClass
属性,它可以获得当前类中的所有函数。