1. 程式人生 > >Spring 框架中切入點 pointcut 表示式的常用寫法

Spring 框架中切入點 pointcut 表示式的常用寫法

自從使用 AspectJ 風格切面配置,使得 Spring 的切面配置大大簡化,但是 AspectJ 是另外一個開源專案,其規則表示式的語法也稍稍有些怪異。

下面給出一些常見示例的寫法,例如,下面是一個對 Service 包上所有方法的切面配置:

<aop:config> 
    <aop:pointcut id="serviceOperation" expression="execution(* *..service*..*(..))"/> 
    <aop:advisor pointcut-ref="serviceOperation" advice-ref
="txAdvice"/>
</aop:config>

表示式所處位置如上pointcut的位置。配置這個是為了更好控制切面上的事務,下面是一個事物配置的簡單例子:

<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>

通過切面、通知的配置,就為所有的以deletesaveupdate開頭的方法新增上了一致性事務,對其他方法新增上了只讀事務。

這個還不夠細,如果要寫更為詳細的控制,就需要研究 AspectJ 切點配置的語法了,其實研究這些標準,還不如拿幾個例子看看,解決實際問題就行了。就像寫正則表示式一樣,標準明擺著,要寫好卻不容易,從例子著手就能快速上手和領悟。

以下文件來自 Spring 中文開發指南 2.5 文件,由滿江紅開源組織翻譯:

Spring AOP 使用者可能會經常使用 execution 切入點指示符。執行表示式的格式如下:

execution(modifiers-pattern? 
ret-type-pattern declaring-type-pattern? 
name-pattern(param-pattern)throws-pattern?

除了返回型別模式(上面程式碼片斷中的 ret-type-pattern),名字模式和引數模式以外, 所有的部分都是可選的。返回型別模式決定了方法的返回型別必須依次匹配一個連線點。 你會使用的最頻繁的返回型別模式是*,它代表了匹配任意的返回型別。 一個全限定的型別名將只會匹配返回給定型別的方法。名字模式匹配的是方法名。你可以使用*萬用字元作為所有或者部分命名模式。 引數模式稍微有點複雜,()匹配了一個不接受任何引數的方法, 而(..)則匹配了一個接受任意數量引數的方法(零或者更多)。模式(*)匹配了一個接受一個任何型別的引數的方法。 模式(*,String)匹配了一個接受兩個引數的方法,第一個可以是任意型別, 第二個則必須是 String 型別。更多的資訊請參閱 AspectJ 程式設計指南中 語言語義的部分。下面給出一些通用切入點表示式的例子。

任意公共方法的執行:

execution(public * *(..))

任何一個名字以 set 開始的方法的執行:

execution(* set*(..))

AccountService 介面定義的任意方法的執行:

execution(* com.xyz.service.AccountService.*(..))

在 service 包中定義的任意方法的執行:

execution(* com.xyz.service.*.*(..))

在 service 包或其子包中定義的任意方法的執行:

execution(* com.xyz.service..*.*(..))

在 service 包中的任意連線點(在 Spring AOP 中只是方法執行):

within(com.xyz.service.*)

在 service 包或其子包中的任意連線點(在 Spring AOP 中只是方法執行):

within(com.xyz.service..*)

實現 AccountService 介面的代理物件的任意連線點 (在 Spring AOP 中只是方法執行):

this(com.xyz.service.AccountService

實現 AccountService 介面的目標物件的任意連線點 (在 Spring AOP 中只是方法執行):

target(com.xyz.service.AccountService

任何一個只接受一個引數,並且執行時所傳入的引數是 Serializable 介面的連線點(在 Spring AOP 中只是方法執行):

args(java.io.Serializable

請注意在例子中給出的切入點不同於execution(* *(java.io.Serializable)),args 版本只有在動態執行時候傳入引數是 Serializable 時才匹配,而 execution 版本在方法簽名中宣告只有一個 Serializable 型別的引數時候匹配。

目標物件中有一個 @Transactional 註解的任意連線點 (在 Spring AOP 中只是方法執行):

@target(org.springframework.transaction.annotation.Transactional

任何一個目標物件宣告的型別有一個 @Transactional 註解的連線點 (在 Spring AOP 中只是方法執行):

@within(org.springframework.transaction.annotation.Transactional

任何一個執行的方法有一個 @Transactional 註解的連線點 (在 Spring AOP 中只是方法執行):

@annotation(org.springframework.transaction.annotation.Transactional

任何一個只接受一個引數,並且執行時所傳入的引數型別具有 @Classified 註解的連線點(在 Spring AOP 中只是方法執行):

@args(com.xyz.security.Classified

任何一個在名為 tradeService 的 Spring bean 之上的連線點 (在 Spring AOP 中只是方法執行):

bean(tradeService)

任何一個在名字匹配萬用字元表示式*Service的 Spring bean 之上的連線點 (在 Spring AOP 中只是方法執行):

bean(*Service

其中,thistagartargs@target@with@annotation@args在繫結表單中更加常用。