隨筆-62  評論-29  文章-0  trackbacks-0

          Spring 2.0 與Spring 1.0 AOP 與事務(wù)配置區(qū)別

          ************Spring 1.0的標(biāo)準(zhǔn)事務(wù)配置***************
          先定義一個baseTxService進行基本的事務(wù)定義,類型為TransactionProxyFactoryBean。如果service沒有基于接口,使用cgilib來實現(xiàn)AOP,定義<property name="proxyTargetClass" value="true"/>
          實際的Manager類設(shè)置parent=baseTxService,target 為匿名的實際Manager類。如果需要定義特殊的事務(wù),利用merge=true的屬性,在manager的transactionAttributes節(jié)點進行新事務(wù)的定義。

          <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來實現(xiàn)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提到的限制,事關(guān)安全acegi methodSecurityInterceptor 攔截器要配置在關(guān)于事務(wù)的TransactionProxyFactoryBean的preInterceptors屬性里,這樣子就一點不AOP了。

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

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

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

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

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

          唯一有點難懂的是pointcut里的語法,其實也很好學(xué),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個參數(shù)或者多個參數(shù)的,任意類型
          (x,..) 第一個參數(shù)的類型必須是X
          (x,,,s,..) 匹配至少4個參數(shù),第一個參數(shù)必須是x類型,第二個和第三個參數(shù)可以任意,第四個必須是s類型。
          注意事項:
          1. name-pattern千萬不要寫成*..*Manager ,這樣子會把所有第三方類庫的Manager比如Spring的PlatformTranstationManager 也加入aop,非常危險。所以最好還是加上項目的package前綴,如"org.springside..*Manager"

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

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

          posted on 2009-09-03 17:59 閱讀(338) 評論(0)  編輯  收藏

          只有注冊用戶登錄后才能發(fā)表評論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 南投市| 石林| 三明市| 新沂市| 监利县| 乌拉特中旗| 将乐县| 吕梁市| 龙井市| 美姑县| 通渭县| 客服| 丰宁| 湘乡市| 文昌市| 昭平县| 贵定县| 库车县| 唐山市| 海淀区| 马龙县| 广汉市| 万安县| 霍林郭勒市| 蒙山县| 济阳县| 桃园市| 平果县| 秦皇岛市| 襄汾县| 沙坪坝区| 休宁县| 龙陵县| 呼和浩特市| 隆昌县| 张家川| 葫芦岛市| 南丹县| 德保县| 镇原县| 榆社县|