Spring切入點表達式經常使用寫法

Spring切入點表達式經常使用寫法
 
自從使用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 >
 
經過切面、通知的配置,就爲全部的delete/save/update開頭的方法添加上了一致性事務,對其餘方法添加上了只讀事務。
 
 
這個還不夠細,若是要寫更爲詳細的控制,就須要研究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)
    'this'在綁定表單中更加經常使用:- 請參見後面的通知一節中瞭解如何使得代理對象在通知體內可用。
  • 實現 AccountService接口的目標對象的任意鏈接點 (在Spring AOP中只是方法執行):
    target(com.xyz.service.AccountService)
    'target'在綁定表單中更加經常使用:- 請參見後面的通知一節中瞭解如何使得目標對象在通知體內可用。
  • 任何一個只接受一個參數,而且運行時所傳入的參數是 Serializable 接口的鏈接點(在Spring AOP中只是方法執行)
    args(java.io.Serializable)
    'args'在綁定表單中更加經常使用:- 請參見後面的通知一節中瞭解如何使得方法參數在通知體內可用。
    請注意在例子中給出的切入點不一樣於 execution(* *(java.io.Serializable)): args版本只有在動態運行時候傳入參數是Serializable時才匹配,而execution版本在方法簽名中聲明只有一個 Serializable類型的參數時候匹配。
  • 目標對象中有一個 @Transactional 註解的任意鏈接點 (在Spring AOP中只是方法執行)
    @target(org.springframework.transaction.annotation.Transactional)
    '@target'在綁定表單中更加經常使用:- 請參見後面的通知一節中瞭解如何使得註解對象在通知體內可用。
  • 任何一個目標對象聲明的類型有一個 @Transactional 註解的鏈接點 (在Spring AOP中只是方法執行):
    @within(org.springframework.transaction.annotation.Transactional)
    '@within'在綁定表單中更加經常使用:- 請參見後面的通知一節中瞭解如何使得註解對象在通知體內可用。
  • 任何一個執行的方法有一個 @Transactional 註解的鏈接點 (在Spring AOP中只是方法執行)
    @annotation(org.springframework.transaction.annotation.Transactional)
    '@annotation'在綁定表單中更加經常使用:- 請參見後面的通知一節中瞭解如何使得註解對象在通知體內可用。
  • 任何一個只接受一個參數,而且運行時所傳入的參數類型具備 @Classified 註解的鏈接點(在Spring AOP中只是方法執行)
    @args(com.xyz.security.Classified)
    '@args'在綁定表單中更加經常使用:- 請參見後面的通知一節中瞭解如何使得註解對象在通知體內可用。
  • 任何一個在名爲' tradeService'的Spring bean之上的鏈接點 (在Spring AOP中只是方法執行):
    bean(tradeService)
  • 任何一個在名字匹配通配符表達式' *Service'的Spring bean之上的鏈接點 (在Spring AOP中只是方法執行):
    bean(*Service)
相關文章
相關標籤/搜索