AOP的基本概念
@Aspect(切面): 通常是一個類灿里,里面可以定義切入點和通知
JointPoint(連接點):? 程序執(zhí)行過程中明確的點武花,一般是方法的調(diào)用
Advice(通知):? AOP在特定的切入點上執(zhí)行的增強處理:
@Before:??標(biāo)識一個前置增強方法德玫,相當(dāng)于BeforeAdvice的功能
@After:? final增強负间,不管是拋出異车溆瑁或者正常退出都會執(zhí)行城看。
@AfterReturning:? 后置增強封救,似于AfterReturningAdvice, 方法正常退出時執(zhí)行
@AfterThrowing:? 異常拋出增強拇涤,相當(dāng)于ThrowsAdvice
@Around: 環(huán)繞增強,相當(dāng)于MethodInterceptor
Pointcut(切入點):? ?帶有通知的連接點誉结,在程序中主要體現(xiàn)為書寫切入點表達(dá)式
AOP Proxy:AOP框架創(chuàng)建的對象鹅士,代理就是目標(biāo)對象的加強。Spring中的AOP代理可以使JDK動態(tài)代理惩坑,也可以是CGLIB代理掉盅,前者基于接口,后者基于子類以舒。
Pointcut配置使用:
表示式(expression)和簽名(signature)
//Pointcut表示式
@Pointcut("execution(* com.savage.aop.MessageSender.*(..))")
//Point簽名
private void log(){}
由下列方式來定義或者通過 &&趾痘、?||、?!蔓钟、?的方式進(jìn)行組合:
execution:用于匹配方法執(zhí)行的連接點永票;
within:用于匹配指定類型內(nèi)的方法執(zhí)行;
this:用于匹配當(dāng)前AOP代理對象類型的執(zhí)行方法;注意是AOP代理對象的類型匹配侣集,這樣就可能包括引入接口也類型匹配键俱;????????
target:用于匹配當(dāng)前目標(biāo)對象類型的執(zhí)行方法;注意是目標(biāo)對象的類型匹配世分,這樣就不包括引入接口也類型匹配编振;
args:用于匹配當(dāng)前執(zhí)行的方法傳入的參數(shù)為指定類型的執(zhí)行方法;
@within:用于匹配所以持有指定注解類型內(nèi)的方法罚攀;
@target:用于匹配當(dāng)前目標(biāo)對象類型的執(zhí)行方法党觅,其中目標(biāo)對象持有指定的注解;
@args:用于匹配當(dāng)前執(zhí)行的方法傳入的參數(shù)持有指定注解的執(zhí)行斋泄;
@annotation:用于匹配當(dāng)前執(zhí)行方法持有指定注解的方法杯瞻;
格式
execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)throws-pattern?)
其中后面跟著“?”的是可選項
括號中各個pattern分別表示:
修飾符匹配(modifier-pattern?)
返回值匹配(ret-type-pattern)可以為*表示任何返回值,全路徑的類名等
類路徑匹配(declaring-type-pattern?)
方法名匹配(name-pattern)可以指定方法名 或者 *代表所有,?set* 代表以set開頭的所有方法
參數(shù)匹配((param-pattern))可以指定具體的參數(shù)類型,多個參數(shù)間用“,”隔開炫掐,各個參數(shù)也可以用"*"?來表示匹配任意類型的參數(shù)魁莉,".."表示零個或多個任意參數(shù)。
如(String)表示匹配一個String參數(shù)的方法募胃;(*,String) 表示匹配有兩個參數(shù)的方法旗唁,第一個參數(shù)可以是任意類型,而第二個參數(shù)是String類型
異常類型匹配(throws-pattern?)
Pointcut使用詳細(xì)語法:
任意公共方法的執(zhí)行:execution(public * *(..))
任何一個以“set”開始的方法的執(zhí)行:execution(* set*(..))
AccountService 接口的任意方法的執(zhí)行:execution(* com.xyz.service.AccountService.*(..))
定義在service包里的任意方法的執(zhí)行:?execution(* com.xyz.service.*.*(..))
定義在service包和所有子包里的任意類的任意方法的執(zhí)行:execution(* com.xyz.service..*.*(..))
第一個*表示匹配任意的方法返回值痹束, ..(兩個點)表示零個或多個检疫,第一個..表示service包及其子包,第二個*表示所有類, 第三個*表示所有方法,第二個..表示方法的任意參數(shù)個數(shù)
定義在pointcutexp包和所有子包里的JoinPointObjP2類的任意方法的執(zhí)行:execution(* com.test.spring.aop.pointcutexp..JoinPointObjP2.*(..))")
pointcutexp包里的任意類:?within(com.test.spring.aop.pointcutexp.*)
pointcutexp包和所有子包里的任意類:within(com.test.spring.aop.pointcutexp..*)
實現(xiàn)了Intf接口的所有類,如果Intf不是接口,限定Intf單個類:this(com.test.spring.aop.pointcutexp.Intf)
當(dāng)一個實現(xiàn)了接口的類被AOP的時候,用getBean方法必須cast為接口類型,不能為該類的類型
帶有@Transactional標(biāo)注的所有類的任意方法:
@within(org.springframework.transaction.annotation.Transactional) // 親測成功
@target(org.springframework.transaction.annotation.Transactional) // 親測未成功
帶有@Transactional標(biāo)注的任意方法:@annotation(org.springframework.transaction.annotation.Transactional)
@within和@target針對類的注解,@annotation是針對方法的注解
參數(shù)帶有@Transactional標(biāo)注的方法:@args(org.springframework.transaction.annotation.Transactional)
參數(shù)為String類型(運行是決定)的方法:?args(String)
JoinPoint 常用的方法:
Object[] getArgs:返回目標(biāo)方法的參數(shù)
Signature getSignature:返回目標(biāo)方法的簽名
Object getTarget:返回被織入增強處理的目標(biāo)對象
Object getThis:返回AOP框架為目標(biāo)對象生成的代理對象
當(dāng)使用@Around處理時祷嘶,需要將第一個參數(shù)定義為ProceedingJoinPoint類型屎媳,該類是JoinPoint的子類。
織入
@AfterReturning(
pointcut="execution(* com.abc.service.*.access*(..)) && args(time, name)",returning="returnValue")
public void access(Date time, Object returnValue, String name){
? ? ? ? System.out.println("目標(biāo)方法中的參數(shù)String = " + name);
? ? ? ? System.out.println("目標(biāo)方法中的參數(shù)Date = " + time);
? ? ? ? System.out.println("目標(biāo)方法的返回結(jié)果returnValue = " + returnValue);
}
表達(dá)式中增加了args(time, name)部分论巍,意味著可以在增強處理的簽名方法(access方法)中定義"time"和"name"兩個屬性烛谊。這兩個形參的類型可以隨意指定,但一旦指定了這兩個參數(shù)的類型嘉汰,則這兩個形參類型將用于限制該切入點只匹配第一個參數(shù)類型為Date丹禀,第二個參數(shù)類型為String的方法(方法參數(shù)個數(shù)和類型若有不同均不匹配);access方法只需要滿足"time", "name"參數(shù)的順序和pointcut中args(param1, param2)的順序相同即可鞋怀,"returnValue"位置順序無所謂双泪。
//將被access方法匹配
public String accessAdvice(Date d, String n){? ??
????System.out.println("方法:accessAdvice");
????return"aa";
}
切面執(zhí)行順序
一個方法只被一個Aspect類攔截
正常:
異常:
同一個方法被多個Aspect類攔截
優(yōu)先級高的切面類里的增強處理的優(yōu)先級總是比優(yōu)先級低的切面類中的增強處理的優(yōu)先級高。
在“進(jìn)入”連接點時接箫,最高優(yōu)先級的增強處理將先被織入(eg.給定的兩個不同切面類Before增強處理中攒读,優(yōu)先級高的那個會先執(zhí)行);在“退出”連接點時辛友,最高優(yōu)先級的增強處理會最后被織入(eg.給定的兩個不同切面類After增強處理中,優(yōu)先級高的那個會后執(zhí)行)。eg.優(yōu)先級為1的切面類Bean1包含了@Before废累,優(yōu)先級為2的切面類Bean2包含了@Around邓梅,雖然@Around優(yōu)先級高于@Before,但由于Bean1的優(yōu)先級高于Bean2的優(yōu)先級邑滨,因此Bean1中的@Before先被織入日缨。
Spring提供了如下兩種解決方案指定不同切面類里的增強處理的優(yōu)先級:
讓切面類實現(xiàn)org.springframework.core.Ordered接口:實現(xiàn)該接口的int getOrder()方法,該方法返回值越小掖看,優(yōu)先級越高
直接使用@Order注解來修飾一個切面類:使用這個注解時可以配置一個int類型的value屬性匣距,該屬性值越小,優(yōu)先級越高
同一個切面類里的兩個相同類型的增強處理在同一個連接點被織入時哎壳,Spring AOP將以隨機的順序來織入這兩個增強處理毅待,沒有辦法指定它們的織入順序。即使給這兩個 advice 添加了 @Order 這個注解归榕,也不行尸红!
eg.
spring的xml配置:
進(jìn)入切面的方法:
關(guān)鍵總結(jié):
帶有@Transactional標(biāo)注的所有類的任意方法:
@within(org.springframework.transaction.annotation.Transactional) // 親測成功
@target(org.springframework.transaction.annotation.Transactional) // 親測未成功
帶有@Transactional標(biāo)注的任意方法:@annotation(org.springframework.transaction.annotation.Transactional)
@within和@target針對類的注解,@annotation是針對方法的注解