我很快就会发布真正的答案,但如果有人想通过反射来做到这一点,代码的外观如下。更复杂,但关于如何将 Kotlin 反射用于 KCallable 的教育。
这是要调用的类:
class Foo {
public fun bar(name: String = "World"): String = "Hello $name!"
}
然后我们需要 Kotin 中的实用程序类,它可以接收类的实例、来自 java 反射的方法以及名称参数。这仅适用于非基元:
class KReflectHelper {
companion object {
@Suppress("UNCHECKED_CAST")
@JvmStatic fun <T> callKotlinMethodWithNamedParms(instance: Any, method: Method, parmMap: Map<String, Any>): T {
val callable: KFunction<T> = method.kotlinFunction as? KFunction<T> ?: throw IllegalStateException("Method is not a Kotlin method")
val unusedParms = HashSet(parmMap.keys)
val callableParms = hashMapOf<KParameter, Any?>()
callable.parameters.map { parm ->
if (parm.kind == KParameter.Kind.INSTANCE) {
callableParms.put(parm, instance)
} else if (parm.kind == KParameter.Kind.VALUE && parmMap.contains(parm.name)) {
unusedParms.remove(parm.name)
callableParms.put(parm, parmMap.get(parm.name))
} else if (parm.kind == KParameter.Kind.VALUE) {
if (parm.isOptional) {
// default value will be used!
} else {
throw IllegalStateException("Missing required parameter ${parm.name}")
}
} else {
throw IllegalStateException("Cannot call methods that are not direct instance methods")
}
}
if (unusedParms.isNotEmpty()) {
throw IllegalStateException("Unrecognized parameters passed to function: $unusedParms")
}
return method.kotlinFunction?.callBy(callableParms) as T
}
}
}
现在可以从 Java 中调用静态方法,但它并没有那么有趣。确实需要代码生成器。从 Kotlin 调用它要容易得多,并且一些框架(例如 Klutter 和 Kovert)已经使用了这些方法。
Foo foo = new Foo();
System.out.println(foo.bar("Frank"));
Method barMethod = Foo.class.getMethod("bar", String.class);
Map<String, Object> parms = new HashMap<String, Object>();
parms.put("name", "David");
System.out.println(KReflectHelper.callKotlinMethodWithNamedParms(foo, barMethod, parms));
// now call using the default
parms.clear();
System.out.println(KReflectHelper.callKotlinMethodWithNamedParms(foo, barMethod, parms));
输出:
你好弗兰克!
大卫你好!
世界你好!