这是使用InvocationHandler
的Kotlin等效的Java代码:
override fun invoke(proxy: Any?, method: Method?, args: Array<out Any>?): Any {
println("before httprequest--->" + args)
val ret = method!!.invoke(obj, args)
println("after httprequest--->")
return ret
}
Java代码:
public Object invoke(Object o, Method method, Object[] args) throws Throwable {
System.out.println("jdk--------->http" + args);
Object result=method.invoke(target, args);
System.out.println("jdk--------->http");
return result;
}
在这两种情况下args
都为空,但如果我运行它,Kotlin代码就会给出异常
Exception in thread "main" java.lang.IllegalArgumentException: wrong number of arguments
Kotlin使用标准Java类的原因是什么?
答案 0 :(得分:6)
当您将args
传递给Kotlin中的method!!.invoke(obj, args)
时,它实际上是数组类型的单个参数,默认情况下它不会作为单独的参数分解为其元素
要改变这种行为,请使用spread operator:*args
val ret = method!!.invoke(obj, *args)
使用此语法,args
将以与Java varargs相同的方式传递。例如,这些代码行是等效的:
someVarargsFunction("a", "b", "c", "d", "e")
someVarargsFunction("a", "b", *arrayOf("c", "d"), "e")
注意:如果方法没有任何参数,args
将为null
,并将其传播到Kotlin将导致{{1} }。要解决此问题,请使用NullPointerException
,并在描述的边案中选择正确的部分并将其传播到零参数中。
我的示例代理实现:
*(args ?: arrayOfNulls<Any>(0))
interface SomeInterface {
fun f(a: Int, b: Int): Int
}
val obj = object : SomeInterface {
override fun f(a: Int, b: Int) = a + b
}
输出是:
val a = Proxy.newProxyInstance( SomeInterface::class.java.classLoader, arrayOf(SomeInterface::class.java)) { proxy, method, args -> println("Before; args: " + args?.contentToString()) val ret = method!!.invoke(obj, *(args ?: arrayOfNulls<Any>(0))) println("After; result: $ret") ret } as SomeInterface println(a.f(1, 2))