18.5.2 动态代理和AOP
开发实际应用的软件系统时,通常会存在相同代码段重复出现
的情况,对于这种情况
- 初级的开发只会复制粘贴这些代码.
- 稍有经验的开发者会将这些重复的代码封装一个方法,然后直接调用该方法即可。但采用这种方式来实现代码复用依然产生一个重要问题:那就是调用者和这个方法耦合了。
最理想的情况是即可以执行这个相同的代码段,又无须调用封装的方法,这时就可以通过动态代理
来达到这种效果。
由于JDK
动态代理只能为接口创建动态代理,所以下面先提供一个Dog
接口,该接口代码非常简单,仅仅在该接口里定义了两个方法。
1 | public interface Dog |
上面接口里只是简单地定义了两个方法,并未提供方法实现。如果直接使用Poxy
为该接口创建动态代理对象,则动态代理对象的所有方法的执行效果又将完全一样。实际情况通常是,软件系统会为该Dog
接口提供一个或多个实现类。此处先提供一个简单的实现类: GunDog
。
1 | public class GunDog implements Dog |
上面代码没有丝毫的特别之处,该Dog
的实现类仅仅为每个方法提供了一个简单实现。
下面提供一个DogUtil
类,该类里包含两个通用方法:
1 | public class DogUtil |
借助于Proxy
和InvocationHandler
就可以实现——当程序调用info()
方法和run()
方法时,系统可以自动”将method1()
和method2()
两个通用方法插入info()
和run()
方法中执行。
这个程序的关键在于下面的MylnvokationHandler
类,该类是一个InvocationHandler
实现类,该实现类的invoke()
方法将会作为代理对象的方法实现。
1 | import java.lang.reflect.*; |
上面程序实现invoke()
方法时包含了一行关键代码,这行代码通过反射以target
作为主调来执行method
方法,这就是回调了target
对象的原有方法。在粗体字代码之前调用DogUtil
对象的method1()
方法,在粗体字代码之后调用DogUti
对象的method2()
方法。
下面再为程序提供一个MyProxyFactory
类,该类对象专为指定的target
生成动态代理实例。
1 | import java.lang.reflect.*; |
上面的动态代理工厂类提供了一个getProxy()
方法,该方法为target
对象生成一个动态代理对象,这个动态代理对象与target
实现了相同的接口,所以具有相同的public
方法——一从这个意义上来看,动态代理对象可以当成target
对象使用。当程序调用动态代理对象的指定方法时,实际上将变为执行MylnvokationHandler
对象的invoke
方法。例如,调用动态代理对象的info
方法,程序将开始执行invoke
方法,其执行步骤如下。
下面提供一个主程序来测试这种动态代理的效果。
1 | public class Test |
上面程序中的dog
对象实际上是动态代理对象,只是该动态代理对象也实现了Dog
接口,所以也可以当成Dog
对象使用。程序执行dog
的info()
和run()
方法时,实际上会先执行DogUtil
的method1()
方法,再执行target
对象的info()
和run()
方法,最后执行DogUtil
的method2()
方法。运行结果如下:
1 | =====模拟第一个通用方法===== |
采用动态代理可以非常灵活地实现解耦。通常而言,使用Proxy
生成一个动态代理时,往往并不会凭空产生一个动态代理,这样没有太大的实际意义。通常都是为指定的目标对象生成动态代理。
这种动态代理在AOP
( Aspect Orient Programming
,面向切面编程)中被称为AOP
代理,AOP
代理可代替目标对象,AOP
代理包含了目标对象的全部方法。但AOP
代理中的方法与目标对象的方法存在差异:AOP
代理里的方法可以在执行目标方法之前、之后插入一些通用处理。
本文重点
JDK
动态代理只能为接口创建动态代理
创建代理对象的步骤:
- 编写自定义
InvocationHandler
实现类,重写invoke()
方法,nvoke
方法的第一个参数proxy
代表要动态代理的对象,第二个参数method
:代表要执行的目标方法第三个参数args
:代表调用目标方法时传入的实参。在invoke()
方法中,在目标方法的前面和后面添加增强的方法. - 编写接口,以及该接口的实现类,然后创建实现类,赋值给接口引用(多态),这样就得到了一个接口的实例对象(目标对象)
- 调用
Proxy.newProxyInstance()
方法生成代理对象,newProxyInstance
方法的第一个参数是要代理的目标对象的类加载器,第二个参数是目标对象的接口,第三个参数是自定义InvocationHandler
实现类。 - 由于代理类和被代理类都实现相同的接口,所以可以调用同名的方法.
原文链接: 18.5.2 动态代理和AOP