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