Spring AOP 基于注解詳解及實例代碼
1.啟用spring對@AspectJ注解的支持:
<beans xmlns:aop="http://www.springframework.org/schema/aop"...> <!--啟動支持--> <aop:aspectj-autoproxy /></beans>
也可以配置AnnotationAwareAspectJAutoProxyCreator Bean來啟動Spring對@AspectJ注解的支持
<beans...> <bean class="org.springframework.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator" /></beans>
2.為了在應用中啟動@AspectJ的支持,還需亞奧增加兩個AspectJ庫:aspectjweaver.jar和aspectjrt.jar。除此之外,Spring AOP還需要依賴一個aopllliance.jar包
3.定義切面Bean
@Aspectpublic class LogAspect { //...}
4.定義增強處理器,如Before
@Aspectpublic class LogAspect { @Before("execution(* *.*.*(..))") public void authority() { System.out.println("執行目標方法前模擬權限檢查") ; }}
@AfterReturning注解將在目標方法正常完成后被織入,該注解指定如下兩個屬性:
1>pointcut/value:用于指定該切入點對應的切入表達式
2>returning:指定一個形參名,用于訪問目標方法的返回值。同時如果在Advice方法中指定該形參類型,將會限制目標方法的返回值必須為該類型
@AfterThrowing注解用于處理程序中未處理的異常。該注解指定如下連個屬性:
1>pointcut/value:用于指定該切入點對應的切入表達式
2>throwing:該屬性值也指定一個形參名,用于表示目標方法拋出的未處理的異常。同時如果在Advice方法中指定該形參類型,將會限制目標方法必須拋出指定類型的異常
@Around注解近似于Before和AfterReturning增強處理的總和,它可以決定目標方法在什么時候執行,因為該注解修飾的Advice方法第一個形參為ProceedingJoinPoint類型,ProceedingJoinPoint參數有一個proceed()方法,調用該方法可以執行目標方法。如果在Advice方法中沒有顯示調用該方法, 那么目標方法將不會被執行:
@Aspectpublic class LogAspect { @Around("execution(* *.*.*(..))") public Object aroundAdvice(ProceedingJoinPoint jp) { System.out.println("執行Around增強處理") ; //獲取目標方法的原始參數 Object[] args = jp.getArgs() ; //執行目標方法獲取返回值 Object result = jp.proceed(args) ; System.out.priontln("Around增強處理執行完畢") ; }}
5.如果需要獲取目標方法的相關信息,可以在定義增強處理方法時將第一個參數定義為JoinPoint類型,當該增強處理方法被調用時,該JoinPoint參數就代表了織入增強處理的連接點。JoinPoint類似與Around增強處理的ProceedingJoinPoint,只不過后者特定于Around增強處理使用。JoinPoint里包含了如下幾個常用的方法:
1>Object[] getArgs():返回執行目標方法時的參數 2>Signature getSignature():返回被增強的方法的相關信息 3>Object getTarget():返回被織入增處理的目標對象 4>Object getThis():返回AOP框架為目標對象生成的代理對象
示例Before增強處理獲取目標方法的相關信息
@Aspectpublic class LogAspect { @Before("execution(* *.*.*(..))") public void beforeAdvice(JoinPoint jp) { //獲取目標方法的參數 Object[] args = jp.getArgs() ; System.out.println("目標方法的參數列表為:" + Array.toString(args)) ; //獲取目標方法的方法名 String methodName = jp.getSignature().getName() ; System.out.println("目標方法的方法名為:" + methodNamme) ; //獲取被織入增強處理的目標對象LogAspect System.out.println("被織入增強處理的目標對象為:" + jp.getTarget()) ; }}
6.如果兩個不同的Aspect里的兩個Advice需要在同一個JoinPoint連接點被植入時,Spring AOP將會以隨機的順序來織入這兩個增強處理。如果需要指定他們的優先級,有兩種方法:
1>Aspect類實現org.springframework.core.Ordered接口,并且實現其抽象方法:int getOrder();該方法的返回值越小,優先級就越高 2>直接使用@Order注解來修飾Aspect類,該注解需要指定一個int型的value屬性值
7.定義切入點Pointcut:包含兩個部分:一個切入點表達式和一個包含名字和任意參數的方法簽名:
@Pointcut("execution(* *.*.*(..))")public void simplepointcut() { //...}//定義了Pointcut之后,就可以在其他的增強處理中使用其方法名作為pointcut屬性值了:@Before(pointcut="simplepointcut()")//等同于pointcut="execution(* *.*.*(..))"public void beforeAspect() { //...}//如果使用的pointcut切入點不是同一類,就需要使用類來修飾如: @Before(pointcut="Simple.simplepointcut()") ...
8.切入點指示符:也就是之前增強處理中指定的execution一類,用于指定目標方法要滿足的條件。Spring AOP一共支持如下幾種切入點指示符:
1>execution:用于匹配執行方法的joinpoint 2>within:用于限定匹配特定類型的joinpoint 如: within(com.cheng.joinpoint.*) 3>this:用于限定AOP代理必須是指定類型的實例,匹配該對象的所有連接點 如: this(com.cheng.aop.aopService) 4>target:用于限定目標獨享必須是指定類型的實例 5>args:用于對連接點的參數類型進行限制,要求參數類型必須是指定類型的實例,多個參數類型使用逗號隔開 6>bean:用于限定只匹配指定Bean的實例內的連接點,需要傳入Bean的id/name
9>最后,我們需要在Spring配置文件中使用元素來指定自動搜索切面類
<beans...> <!--指定自動搜索Bean組件、自動搜索切面類--> <context:component-scan base-package="com.cheng"> <context:include-filter type="annotation" expression="org.aspectj.lang.annotation.Aspect" /> </context:component-scan></beans>
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
新聞熱點
疑難解答