什么是动态代理
在本文中,讨论的是kotlin中如何使用类似于JDK动态代理的功能,而非kotlin中by关键字实现的代理模式,所以可以说此文是介绍如何用kotlin实现JDK动态代理,至于JDK动态代理是什么,本文不予赘述,读者可自行查阅有关资料。
如何使用
在正式开始使用前,先介绍下几个比较重要的东西
- 在动态代理中,最重要的是
InvocationHandler
,因为最终实现代理就是在InvocationHandler
的invoke
方法中实现的,InvocationHandler
是一个接口,该接口中只有一个invoke
方法。 - 要被代理的类必须继承自一个接口,换句话说,要被代理的方法,必须利用接口进行声明。
下面开始正片:
- 为需要被代理的类及方法创建接口
interface IDemo{ fun test(action: String, code: Int) }
- 创建实现接口的类
class Demo : IDemo{ override fun test(action: String, code: Int) { println("action = [${action}], code = [${code}]") } }
- 进行代理
fun proxyTest() { val demo = Demo() // 创建要被代理的实例 val proxy = Proxy.newProxyInstance( // 获取实例的代理对象 Demo::class.java.classLoader, // 获取实例的classloader arrayOf(IDemo::class.java), // 实例接口 object : InvocationHandler { override fun invoke(proxy: Any, method: Method, args: Array<out Any>): Any { println("before...") // 代理增强方法 val result = method.invoke(demo, *args) // 反射调用实例的原始方法 if ("void" == method.genericReturnType.typeName) // 如果返回值为void, 要转换为Unit, 否则会报空指针异常 return Unit return result } }) as IDemo // 类型转换 proxy.test("hello", 123456) // 调用代理 }
- 输出
before... action = [hello], code = [123456]
坑点
- kotlin直接生成的
InvocationHandler
,其invoke
方法参数类型都是可空的,这里需要手动调整为非空,就是去掉参数类型声明后的?,一般来说这么做是没有问题的,可以参照JDK的InvocationHandler
,当然你也可以进行非空判断,至于kotlin为什么将invoke
方法的形参类型都变成了可空类型,我也不太清楚,有了解的朋友可以评论区科普一下。 - 返回类型,如果被代理方法的返回类型为
void
,那么需要在invoke
方法中返回Unit
Proxy
的newProxyInstance
第二个参数必须为接口的class数组,不能是被代理对象的class,否则会报is not interface
的异常。
最后,这篇文章属于个人经验,可能存在盲点,欢迎各位指出文中的错误之处!