Spring 2.0 AOP 與事務配置

Spring 2.0 AOP 與事務配置
************Spring 1.0的標準事務配置***************
先定義一個baseTxService進行基本的事務定義,類型爲TransactionProxyFactoryBean。如果service沒有基於接口,使用cgilib來實現AOP,定義<property name="proxyTargetClass" value="true"/>
實際的Manager類設置parent=baseTxService,target 爲匿名的實際Manager類。如果需要定義特殊的事務,利用merge=true的屬性,在manager的transactionAttributes節點進行新事務的定義。

<bean id="baseTxService" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"
abstract="true">
<property name="transactionManager" ref="transactionManager"/>
<property name="proxyTargetClass" value="true"/>
<property name="transactionAttributes">
<props>
<prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
<prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>
<prop key="save*">PROPAGATION_REQUIRED</prop>
<prop key="remove*">PROPAGATION_REQUIRED</prop>
</props>
</property>
<property name="preInterceptors">
<list>
<ref bean="methodSecurityInterceptor"/>
</list>
</property>
</bean> <bean id="bookManager" parent="baseTxService">
<property name="target">
<bean class="org.springside.bookstore.admin.manager.BookManager"/>
</property>
</bean>
**************Spring 2.0的新配置:*********
如果如果service沒有基於接口,使用cgilib來實現AOP,定義proxy-target-class="true"

<aop:config proxy-target-class="true">
<aop:advisor pointcut="execution(* com.xyz.service..*Manager.*(..))" advice-ref="txAdvice"/>
<aop:advisor pointcut="execution(* com.xyz.service..*Manager.save(..))" advice-ref="fooAdvice"/>
</aop:config><tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="get*" read-only="true"/>
<tx:method name="find*" read-only="true"/>
<tx:method name="*" />
</tx:attributes>
</tx:advice>

<bean id="bookManager" class="org.springside.bookstore.commons.service.BookManager"/> 其中com.xyz是你的項目package限定前綴。

2.進步
1. AOP的配置方式也AOP了。
對比1.0的配置文件,因爲下面2提到的限制,事關安全acegi methodSecurityInterceptor 攔截器要配置在關於事務的TransactionProxyFactoryBean的preInterceptors屬性裏,這樣子就一點不AOP了。

而2.0使用ponintcut expression,很AOP的配置一切Aspect。

2. 1.0時,一個已經AOP過的object不能再次被AOP。
在Spring 1.0的文檔裏Rod說,比如<bean id="bookManager" parent="baseTxService">已經進行了一次AOP,如果想在這個Bean上再配一層AOP,比如要對方法執行結果緩存,無論以1.0 還是2.0的方式定義,cglib方式是會報錯的,而基於接口的方式,結果不確定。

3. BookManager能直接定義自己,而不是像1.0那樣作匿名內部target。

雖然在1.0時代的BeanNameAutoProxyCreator 達到類似作用,但只能用BeanName來模糊匹配比較危險,沒有AspectJ的pointcut語法細緻。

3. 語法
滿江紅翻譯的 Spring參考文檔 6.3 schema-based AOP support 提供了aspect,advisor,advide三種組裝方法的解釋,其中aspect是aspectJ原裝,但稍複雜。

唯一有點難懂的是pointcut裏的語法,其實也很好學,Spring參考文檔6.2.3.4 示例裏有完整說明 ,其實一排子過去是

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)其中帶問號的modifiers-pattern?(public/protected) 和 declaring-type-pattern? throws-pattern? 可以不填

execution(* *..BookManager.save(..))的解讀:
第一顆* 代表ret-type-pattern 返回值可任意,
*..BookManager 代表任意Pacakge裏的BookManager類。
如果寫成com.xyz.service.* 則代表com.xyz.service下的任意類
com.xyz.service..* com.xyz.service則代表com.xyz.service及其子package下的任意類
save代表save方法,也可以寫save* 代表saveBook()等方法
(..) 匹配0個參數或者多個參數的,任意類型
(x,..) 第一個參數的類型必須是X
(x,,,s,..) 匹配至少4個參數,第一個參數必須是x類型,第二個和第三個參數可以任意,第四個必須是s類型。
注意事項:
1. name-pattern千萬不要寫成*..*Manager ,這樣子會把所有第三方類庫的Manager比如Spring的PlatformTranstationManager 也加入aop,非常危險。所以最好還是加上項目的package前綴,如"org.springside..*Manager"

2. 因爲有*,會修飾所有方法,有些hibernateTemplate的final的方法不能被cglib修改,會拋warning,無害。

4. 事務定義選項
事務定義一般默認的PROPAGATION_REQUIRED即可,另提供的幾個選擇很少使用。值得注意的是一個PROPAGATION_NESTED,嵌入式事務的意義在於多級事務,如果出錯只rollback子事務自己,不rollback主事務的所有操作。比如OrderManager的shipOrder函數 調用 save函數,如果save()被定義爲嵌入式事務,當進入save()時,會存儲save point。如果在save 中出錯,會rollback 到剛纔的save point,但不影響其他的操作。這需要JDBC3.0 SavePoint功能的支持。 而一般service間互相嵌入調用時,如果都定義爲PROPAGATION_REQUIRED,有其中一個操作出錯,rollback全部操作。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章