您好,登錄后才能下訂單哦!
自從使用 AspectJ 風(fēng)格切面配置,使得 spring 的切面配置大大簡化,但是 AspectJ 是另外一個(gè)開源項(xiàng)目,其規(guī)則表達(dá)式的語法也稍稍有些怪異。
下面給出一些常見示例的寫法,例如,下面是一個(gè)對 Service 包上所有方法的切面配置:
<aop:config> <aop:pointcut id="serviceOperation" expression="execution(* *..service*..*(..))"/> <aop:advisor pointcut-ref="serviceOperation" advice-ref="txAdvice"/> </aop:config>
表達(dá)式所處位置如上pointcut的位置。配置這個(gè)是為了更好控制切面上的事務(wù),下面是一個(gè)事物配置的簡單例子:
<tx:advice id="txAdvice" transaction-manager="transactionManager"> <tx:attributes> <tx:method name="delete*" rollback-for="Exception"/> <tx:method name="save*" rollback-for="Exception"/> <tx:method name="update*" rollback-for="Exception"/> <tx:method name="*" read-only="true" rollback-for="Exception"/> </tx:attributes> </tx:advice>
通過切面、通知的配置,就為所有的以delete、save和update開頭的方法添加上了一致性事務(wù),對其他方法添加上了只讀事務(wù)。
這個(gè)還不夠細(xì),如果要寫更為詳細(xì)的控制,就需要研究 AspectJ 切點(diǎn)配置的語法了,其實(shí)研究這些標(biāo)準(zhǔn),還不如拿幾個(gè)例子看看,解決實(shí)際問題就行了。就像寫正則表達(dá)式一樣,標(biāo)準(zhǔn)明擺著,要寫好卻不容易,從例子著手就能快速上手和領(lǐng)悟。
以下文檔來自 Spring 中文開發(fā)指南 2.5 文檔,由滿江紅開源組織翻譯:
Spring AOP 用戶可能會(huì)經(jīng)常使用 execution 切入點(diǎn)指示符。執(zhí)行表達(dá)式的格式如下:
execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)throws-pattern?)
除了返回類型模式(上面代碼片斷中的 ret-type-pattern),名字模式和參數(shù)模式以外, 所有的部分都是可選的。返回類型模式?jīng)Q定了方法的返回類型必須依次匹配一個(gè)連接點(diǎn)。 你會(huì)使用的最頻繁的返回類型模式是*,它代表了匹配任意的返回類型。 一個(gè)全限定的類型名將只會(huì)匹配返回給定類型的方法。名字模式匹配的是方法名。你可以使用*通配符作為所有或者部分命名模式。 參數(shù)模式稍微有點(diǎn)復(fù)雜,()匹配了一個(gè)不接受任何參數(shù)的方法, 而(..)則匹配了一個(gè)接受任意數(shù)量參數(shù)的方法(零或者更多)。模式(*)匹配了一個(gè)接受一個(gè)任何類型的參數(shù)的方法。 模式(*,String)匹配了一個(gè)接受兩個(gè)參數(shù)的方法,第一個(gè)可以是任意類型, 第二個(gè)則必須是 String 類型。更多的信息請參閱 AspectJ 編程指南中 語言語義的部分。下面給出一些通用切入點(diǎn)表達(dá)式的例子。
任意公共方法的執(zhí)行:
execution(public * *(..))
任何一個(gè)名字以 set 開始的方法的執(zhí)行:
execution(* set*(..))
AccountService 接口定義的任意方法的執(zhí)行:
execution(* com.xyz.service.AccountService.*(..))
在 service 包中定義的任意方法的執(zhí)行:
execution(* com.xyz.service.*.*(..))
在 service 包或其子包中定義的任意方法的執(zhí)行:
execution(* com.xyz.service..*.*(..))
在 service 包中的任意連接點(diǎn)(在 Spring AOP 中只是方法執(zhí)行):
within(com.xyz.service.*)
在 service 包或其子包中的任意連接點(diǎn)(在 Spring AOP 中只是方法執(zhí)行):
within(com.xyz.service..*)
實(shí)現(xiàn) AccountService 接口的代理對象的任意連接點(diǎn) (在 Spring AOP 中只是方法執(zhí)行):
this(com.xyz.service.AccountService)
實(shí)現(xiàn) AccountService 接口的目標(biāo)對象的任意連接點(diǎn) (在 Spring AOP 中只是方法執(zhí)行):
target(com.xyz.service.AccountService)
任何一個(gè)只接受一個(gè)參數(shù),并且運(yùn)行時(shí)所傳入的參數(shù)是 Serializable 接口的連接點(diǎn)(在 Spring AOP 中只是方法執(zhí)行):
args(java.io.Serializable)
請注意在例子中給出的切入點(diǎn)不同于execution(* *(Java.io.Serializable)),args 版本只有在動(dòng)態(tài)運(yùn)行時(shí)候傳入?yún)?shù)是 Serializable 時(shí)才匹配,而 execution 版本在方法簽名中聲明只有一個(gè) Serializable 類型的參數(shù)時(shí)候匹配。
目標(biāo)對象中有一個(gè) @Transactional 注解的任意連接點(diǎn) (在 Spring AOP 中只是方法執(zhí)行):
@target(org.springframework.transaction.annotation.Transactional)
任何一個(gè)目標(biāo)對象聲明的類型有一個(gè) @Transactional 注解的連接點(diǎn) (在 Spring AOP 中只是方法執(zhí)行):
@within(org.springframework.transaction.annotation.Transactional)
任何一個(gè)執(zhí)行的方法有一個(gè) @Transactional 注解的連接點(diǎn) (在 Spring AOP 中只是方法執(zhí)行):
@annotation(org.springframework.transaction.annotation.Transactional)
任何一個(gè)只接受一個(gè)參數(shù),并且運(yùn)行時(shí)所傳入的參數(shù)類型具有 @Classified 注解的連接點(diǎn)(在 Spring AOP 中只是方法執(zhí)行):
@args(com.xyz.security.Classified)
任何一個(gè)在名為 tradeService 的 Spring bean 之上的連接點(diǎn) (在 Spring AOP 中只是方法執(zhí)行):
bean(tradeService)
任何一個(gè)在名字匹配通配符表達(dá)式*Service的 Spring bean 之上的連接點(diǎn) (在 Spring AOP 中只是方法執(zhí)行):
bean(*Service)
其中,this
、tagart
、args
、 @target
、 @with
、 @annotation
和@args
在綁定表單中更加常用。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。