相关文章推荐
纯真的苦瓜  ·  自动为DEV ...·  1 年前    · 
腼腆的蛋挞  ·  Mybatis中oracle、mysql的l ...·  1 年前    · 


1 动态代理

动态代理的实现方式常用的有两种:

  • 反射实现:基于反射实现,目标对象必须要有接口。生成的代理类是接口的一个实现类
  • Cglib实现:基于字节码实现,效率稍低,但目标对象不需要有接口。生成的代理类是目标类的子类,因此目标类不能是final的。

2 背景介绍

在Java中java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,通过使用这个类和接口就可以生成动态代理对象,只能针对接口做代理。

接口和目标类没有任何变化,只不过为了增加难度,我们增加了返回值。

接口如下,除了增加了方法的返回值外无任何变化:




Android 反射 aidl参数 反射invoke的参数_代理类 invoke方法没有返回值怎么办


目标类如下,无任何变化:


Android 反射 aidl参数 反射invoke的参数_目标对象_02


2 反射调用方法

接下来,我们使用JDK完成动态代理,对于代理类只需要实现InvocationHandler接口即可,并且要实现其中的invoke()方法。

invoke()中需要传入目标对象、目标对象的方法、调用目标所需的参数,传入后,就会直接调用指定对象的指定方法了。


Android 反射 aidl参数 反射invoke的参数_代理类_03


我们使用举例:


Android 反射 aidl参数 反射invoke的参数_代理类 invoke方法没有返回值怎么办_04


我们只要创建一个ProxyHandler对象,直接调用invoke方法,传入其中所需要的对象、方法、参数,便可以执行我们写好的invoke方法,并且invoke中还可以调用指定对象的指定方法。这种方法非常灵活,ProxyHandler建立后,调用invoke方法时,可以传入任何对象、任何方法,非常自由。

这种操作只是通过反射调用的指定对象的指定方法,并没有使用代理。但是可以用它来实现动态代理。我们只要生成一个代理对象,接口所有的操作都由代理对象来完成。其中,代理对象中再通过这种反射的方式去调用目标对象即可了。

3 动态代理

在使用动态代理时,ProxyHandler需要稍加改造,即在构造方法中传入目标对象。但是这里要注意的是,与静态代理不同,这里的目标对象的类型是任意的,因此同一个ProxyHandler可以用来代理不同的目标对象。


Android 反射 aidl参数 反射invoke的参数_代理类 invoke方法没有返回值怎么办_05


在使用时,如下:


Android 反射 aidl参数 反射invoke的参数_目标对象_06


首先,我们定义了被代理的类的实例,并且传给了proxyHandler。然后,我们我们直接调用了Proxy的静态方法newProxyInstance生成了一个接口的实例,这个实例就是代理(而且它知道它代理的是user对象,初始化时已经传给它了)。

然后,直接对这个实例调用方法就可以了,对这个接口调用方法,就相当于执行了invoke中的操作,输出如下:

pre wordshello Lilypost words

注意,在这种实现中,目标类target是在初始化时写死,并且在invoke时直接调用的,如果在invoke中直接调用传来的proxy对象,则会引发循环调用,造成死循环。

我们查看下Proxy类中的静态方法,newProxyInstance方法:


Android 反射 aidl参数 反射invoke的参数_目标对象_07


该方法用来返回指定接口的一个代理类实例。

输入参数有:

  • 生成代理类实例的classLoader;
  • 要满足的接口;
  • 代理类的处理器,即具体的实现。

这样就指明了用哪个处理器处理哪个接口中的操作,以及该对象如何生成出来。

生成的对象就是一个代理对象,就可以代理接口的操作了。