什么是AOP?
在運(yùn)行時(shí),動(dòng)態(tài)的將代碼切入到類的指定方法、指定位置上的編程思想就是面向切面的編程。AOP即面向切面編程。使用切面編程,可以將一些系統(tǒng)性的代碼提取出來,獨(dú)立實(shí)現(xiàn),與核心業(yè)務(wù)代碼剝離,比如權(quán)限管理、事務(wù)管理、日志記錄等等。AOP是spring提供的關(guān)鍵特性之一。
AOP的實(shí)現(xiàn)原理
AOP分為靜態(tài)AOP和動(dòng)態(tài)AOP,靜態(tài)AOP是由AspectJ實(shí)現(xiàn)的AOP,動(dòng)態(tài)AOP是指將切面代碼進(jìn)行動(dòng)態(tài)織入實(shí)現(xiàn)的AOP,Spring的AOP為動(dòng)態(tài)AOP。實(shí)現(xiàn)的技術(shù)為:JDK提供的動(dòng)態(tài)代理技術(shù)和CGLIB(動(dòng)態(tài)字節(jié)碼增強(qiáng)技術(shù))兩種。盡管實(shí)現(xiàn)技術(shù)不一樣,但都是基于代理模式,都是生成一個(gè)代理對(duì)象。
JDK動(dòng)態(tài)代理實(shí)現(xiàn):查看我前面的博文-動(dòng)態(tài)代理,我們看到該方式有一個(gè)要求, 被代理的對(duì)象必須實(shí)現(xiàn)接口,而且只有接口中的方法才能被代理 。也就是jdk代理方式是基于接口的一種代理方式。
CGLIB:字節(jié)碼生成技術(shù)實(shí)現(xiàn)AOP,其實(shí)就是繼承被代理對(duì)象,然后Override需要被代理的方法,在覆蓋該方法時(shí),自然是可以插入我們自己的代碼的。
因?yàn)樾枰狾verride被代理對(duì)象的方法,所以自然CGLIB技術(shù)實(shí)現(xiàn)AOP時(shí),就必須要求需要被代理的方法不能是final方法,因?yàn)閒inal方法不能被子類覆蓋。
我們使用CGLIB實(shí)現(xiàn)上面的例子:
package net.aazj.aop;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class CGProxy implements MethodInterceptor{
private Object target; // 被代理對(duì)象
public CGProxy(Object target){
this.target = target;
}
public Object intercept(Object arg0, Method arg1, Object[] arg2, MethodProxy proxy) throws Throwable {
System.out.println("do sth before....");
Object result = proxy.invokeSuper(arg0, arg2);
System.out.println("do sth after....");
return result;
}
public Object getProxyObject() {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(this.target.getClass()); // 設(shè)置父類
// 設(shè)置回調(diào)
enhancer.setCallback(this); // 在調(diào)用父類方法時(shí),回調(diào) this.intercept()
// 創(chuàng)建代理對(duì)象
return enhancer.create();
}
}
public class CGProxyTest {
public static void main(String[] args){
Object proxyedObject = new UserServiceImpl(); // 被代理的對(duì)象
CGProxy cgProxy = new CGProxy(proxyedObject);
UserService proxyObject = (UserService) cgProxy.getProxyObject();
proxyObject.getUser(1);
proxyObject.addUser(new User());
}
}
觀察spring中AOP相關(guān)源碼可知,如果被代理對(duì)象實(shí)現(xiàn)了接口,那么就使用JDK的動(dòng)態(tài)代理技術(shù),反之則使用CGLIB來實(shí)現(xiàn)AOP,所以Spring默認(rèn)是使用JDK的動(dòng)態(tài)代理技術(shù)實(shí)現(xiàn)AOP的。
AOP在spring中的使用
Spring中AOP的配置一般有兩種方法,一種是使用 <aop:config> 標(biāo)簽在xml中進(jìn)行配置,一種是使用注解以及@Aspect風(fēng)格的配置。
1)基于<aop:config>的AOP配置
<tx:advice id="transactionAdvice" transaction-manager="transactionManager"?>
<tx:attributes >
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="append*" propagation="REQUIRED" />
<tx:method name="insert*" propagation="REQUIRED" />
<tx:method name="save*" propagation="REQUIRED" />
<tx:method name="update*" propagation="REQUIRED" />
<tx:method name="get*" propagation="SUPPORTS" />
<tx:method name="find*" propagation="SUPPORTS" />
<tx:method name="load*" propagation="SUPPORTS" />
<tx:method name="search*" propagation="SUPPORTS" />
<tx:method name="*" propagation="SUPPORTS" />
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut id="transactionPointcut" expression="execution(* net.aazj.service..*Impl.*(..))" />
<aop:advisor pointcut-ref="transactionPointcut" advice-ref="transactionAdvice" />
</aop:config>
<bean id="aspectBean" class="net.aazj.aop.DataSourceInterceptor"/>
<aop:config>
<aop:aspect id="dataSourceAspect" ref="aspectBean">
<aop:pointcut id="dataSourcePoint" expression="execution(public * net.aazj.service..*.getUser(..))" />
<aop:pointcut expression="" id=""/>
<aop:before method="before" pointcut-ref="dataSourcePoint"/>
<aop:after method=""/>
<aop:around method=""/>
</aop:aspect>
<aop:aspect></aop:aspect>
</aop:config>
<aop:aspect> 配置一個(gè)切面;
<aop:pointcut>配置一個(gè)切點(diǎn),基于切點(diǎn)表達(dá)式;
<aop:before>,<aop:after>,<aop:around>是定義不同類型的advise. a
spectBean 是切面的處理bean
2)基于注解和@Aspect風(fēng)格的AOP配置
參考鏈接:面試題思考:解釋一下什么叫AOP(面向切面編程)
Spring源碼--AOP實(shí)現(xiàn)(1)--創(chuàng)建AopProxy代理對(duì)象
Spring源碼--AOP實(shí)現(xiàn)(2)--攔截器調(diào)用的實(shí)現(xiàn)
總結(jié)
- 上一篇: xfce4快捷键设置
- 下一篇: 基于visual Studio2013解