一、AOP 概念
OC nQSkj Joinpoint:它定义在哪里加入你的逻辑功能,对于Spring AOP,Jointpoint指的就是Method。
bX6eNk-L >'iXwe- Advice:特定的Jointpoint处运行的代码,对于Spring AOP 来讲,有Before advice、AfterreturningAdvice、ThrowAdvice、AroundAdvice(MethodInteceptor)等。
yw `w6Z3K
eU&[^ Pointcut:一组Joinpoint,就是说一个Advice可能在多个地方织入,
@;_xFL;{g V|e9G,z~A Aspect:这个我一直迷惑,它实际是Advice和Pointcut的组合,但是Spring AOP 中的Advisor也是这样一个东西,但是Spring中为什么叫Advisor而不叫做Aspect。
J.W0F# ? &}_ $@ Weaving:将Aspect加入到程序代码的过程,对于Spring AOP,由ProxyFactory或者ProxyFactoryBean负责织入动作。
bW\OKI1 HJ?p,V q5_ Target:这个很容易理解,就是需要Aspect功能的对象。
[v( \y KV0M^B|W Introduction:引入,就是向对象中加入新的属性或方法,一般是一个实例一个引用对象。当然如果不引入属性或者引入的属性做了线程安全性处理或者只读属性,则一个Class一个引用也是可以的(自己理解)。Per-class lifecycle or per-instance life cycle
V]dzKNFi R".~{6 二、AOP 种类
VRQ'sn@ 1、静态织入:指在编译时期就织入Aspect代码,AspectJ好像是这样做的。
ad+@2-Y R8O;8c?D 2、动态织入:在运行时期织入,Spring AOP属于动态织入,动态织入又分静动两种,静则指织入过程只在第一次调用时执行;动则指根据代码动态运行的中间状态来决定如何操作,每次调用Target的时候都执行(性能较差)。
#\3(rzQVO L9ECF;) 三、Spring AOP 代理原理
HJWk%t< Spring AOP 是使用代理来完成的,Spring 会使用下面两种方式的其中一种来创建代理:
*WdnP.'Y 1、JDK动态代理,特点只能代理接口,性能相对较差,需要设定一组代理接口。
d|#sgGM<8 2、CGLIB 代理,可代理接口和类(final method除外),性能较高(生成字节码)。
@ px2/x =$xxkc.~G BQ!v\1'C 四、Spring AOP 通知类型
=kFZ2/P2t( 1、BeforeAdvice:前置通知需实现MethodBeforeAdvice,但是该接口的Parent是BeforeAdvice,致于什么用处我想可能是扩展性需求的设计吧。或者Spring未来也并不局限于Method的JoinPoint(胡乱猜测)。BeforeAdvice可以修改目标的参数,也可以通过抛出异常来阻止目标运行。
O(q1R#n-}+ Xy/lsaVskX 2、AfterreturningAdvice:实现AfterreturningAdvice,我们无法修改方法的返回值,但是可以通过抛出异常阻止方法运行。
<*k]Aa3y Mtxn@m{i;" 3、AroundAdvice:Spring 通过实现MethodInterceptor(aopalliance)来实现包围通知,最大特点是可以修改返回值,当然它在方法前后都加入了自己的逻辑代码,因此功能异常强大。通过MethodInvocation.proceed()来调用目标方法(甚至可以不调用)。
3H"bivK 6?\X)qBI 4、ThrowsAdvice:通过实现若干afterThrowing()来实现。
|:)UNb?R"O ukNB#2" 5、IntroductionInterceptor:Spring 的默认实现为DelegatingIntroductionInterceptor
Q#!|h:K >F7v'-*{ 五、Spring AOP Pointcut
En-BT0o 以上只是Advice,如果不指定切入点,Spring 则使用所有可能的Jointpoint进行织入(当然如果你在Advice中进行方法检查除外)。因此切入点在AOP中扮演一个十分重要的角色。Spring 2.0 推荐使用AspectJ的Annocation的切入点表达式来定义切入点,或者使用<aop:xxx/>来定义AOP,这方面本篇不做考虑。
#$k6OlK-r" )~@iM.}S2 1、Pointcut:它是Spring AOP Pointcut的核心,定义了getClassFilter()和getMethodMatcher()两个方法。
<E/"v JR>v 2、ClassFilter:定义了matches(Class cls)一个方法。
VaKBS/y" .$&_fUY 3、MethodMatcher() 定义了matches(Method,Class),isRuntime(),matches(Mathod,Class,Object[])三个方法,如果isRuntime()返回true则表示为动态代理(实际是动态代理的动态代理),则调用第三个方法(每访问一次调用一次),否则调用第一个方法(并且只调用一次)
5}-e9U (rF XzCI 4、Spring AOP 静态切入点的几个实现。
Nr6[w|Tzd ComposablePointcut 太复杂一个切入点无法表达就用这个,union MethodMatcher和ClassFilter或者intersection MethodMatcher、ClassFilter和Pointcut。为什么不实现union Pointcut? 而只能通过Pointcuts类对Pointcut进行union操作。
Z'/sZ3Q} L{)*evBL ControlFlowPointcut 想对程序的运行过程进行追踪就用这个
1=NP=ZB &.D#OnRh9 DynamicMatchMatcherPointcut 想用动态AOP 就用这个
-bJC+Yn |, ws 3 JdkRegexpMethodPointcut 想使用正则表达式就用这个
*Y"j 0Yob )GgO=J:o Perl5RegexpMethodPointcut
`]*BDSvE slPLc NameMatchMethodPointcut 想用方法名字来匹配就用这个
o]gS=iLp #0*OkZMt StaticMethodMatcherPointcut 静态切入点就用这个
CEOD$nYc 没有人反对你直接实现Pointcut:)。
pXL@&]U+ b&iJui"7k 六、Spring AOP 中的Advisor其实就是Aspect
22`N(_ 1、 PointcutAdvisor
@-d0~.S 其实一般使用DefaultPointcutAdvisor就足够了,给它Advice和Pointcut。
h
7x_VO 当然如果想少写那么几行代码也可以使用NameMatchMethodPointcutAdvisor,RegexpMethodPointcutAdvisor等。
A08b=S 更多Advisor可以查看API文档。
@ckOLtxE> d]bM,`K* 6 TOUP.,f/! 2、 IntroductionAdvisor
*U{E[<k{ 默认实现为DefaultIntroductionAdvisor。
7E4Xvg+c Qx{[#[Da 七、AOP ProxyFactory
]Sl]G6#Iwv 使用代码实现AOP 可使用ProxyFactory
a
d,0*(</ 声明式AOP 可使用ProxyFactoryBean
X!'nfN ProxyFactoryBean 需要设定 target,interceptorNames(可以是Advice或者Advisor,注意顺序)
6e%|.}U 对接口代理需设置proxyInterfaces
(K"8kQLY /d/Quro 八、自动代理
u=mJI* BeanNameAutoProxyCreator
{\87]xJ - <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
- <property name="beanNames"><value>jdk*,onlyJdk</value></property>
- <property name="interceptorNames">
- <list>
- <value>myInterceptor</value>
- </list>
- </property>
- </bean>
c<T'_93 DefaultAdvisorAutoProxyCreator
%\uEV - <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"/>
- <bean class="org.springframework.transaction.interceptor.TransactionAttributeSourceAdvisor">
- <property name="transactionInterceptor" ref="transactionInterceptor"/>
- </bean>
- <bean id="customAdvisor" class="com.mycompany.MyAdvisor"/>
- <bean id="businessObject1" class="com.mycompany.BusinessObject1">
- <!-- Properties omitted -->
- </bean>
- <bean id="businessObject2" class="com.mycompany.BusinessObject2"/>
v<2B^(i}VB Xfq]vQ/{ ab5uZ0@