Spring的兩種事務定義方式
一、宣告式
這種方法不需要對原有的業務做任何修改,通過在XML檔案中定義需要攔截方法的匹配即可完成配置,要求是,業務處理中的方法的命名要有規律,比如setXxx,xxxUpdate等等。詳細配置如下:
- <beanid="transactionManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
-
<propertyname="dataSource"ref="dataSource"
- </bean>
- <!-- 定義事務通知 -->
- <tx:adviceid="txAdvice"transaction-manager="transactionManager">
- <!-- 定義方法的過濾規則 -->
- <tx:attributes>
- <!-- 所有方法都使用事務 -->
- <tx:methodname="*"propagation="REQUIRED"/>
-
<!-- 定義所有get開頭的方法都是隻讀的 -->
- <tx:methodname="get*"read-only="true"/>
- </tx:attributes>
- </tx:advice>
- <!-- 定義AOP配置 -->
- <aop:config>
- <!-- 定義一個切入點 -->
- <aop:pointcutexpression="execution (* com.test.services.impl.*.*(..))"id="services"/>
-
<!-- 對切入點和事務的通知,進行適配 -->
- <aop:advisoradvice-ref="txAdvice"pointcut-ref="services"/>
- </aop:config>
二、註解式
這種方法,只需要在spring配置檔案中定義一個事務管理物件(如DataSourceTransactionManager),然後加入<tx:annotation-driven/>節點,引用該事務管理物件,然後即可在需要進行事務處理的類和方法使用@Transactional進行標註。示例如下:
- <beanid="transactionManager"class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
- <propertyname="dataSource"ref="dataSource"></property>
- </bean>
- <!-- 宣告使用註解式事務 -->
- <tx:annotation-driventransaction-manager="transactionManager"/>
三、<tx:annotation-driven/>
Spring提供的<tx:annotation-driven/>用於開啟對註解事務管理的支援,從而能識別Bean類上的@Transactional註解元資料,其具有以下屬性:
1、<tx:annotation-driven/>
transaction-manager:指定事務管理器名字,預設為transactionManager,當使用其他名字時需要明確指定;
proxy-target-class:表示將使用的程式碼機制,預設false表示使用JDK代理,如果為true將使用CGLIB代理
order:定義事務通知順序,預設Ordered.LOWEST_PRECEDENCE,表示將順序決定權交給AOP來處理。
2、@Transaction
Spring使用@Transaction來指定事務屬性,可以在介面、類或方法上指定,如果類和方法上都指定了@Transaction,則方法上的事務屬性被優先使用,具體屬性如下:
value:指定事務管理器名字,預設使用<tx:annotation-driven/>指定的事務管理器,用於支援多事務管理器環境;
propagation:指定事務傳播行為,預設為Required,使用Propagation.REQUIRED指定;
isolation:指定事務隔離級別,預設為“DEFAULT”,使用Isolation.DEFAULT指定;
readOnly:指定事務是否只讀,預設false表示事務非只讀;
timeout:指定事務超時時間,以秒為單位,預設-1表示事務超時將依賴於底層事務系統;
rollbackFor:指定一組異常類,遇到該類異常將回滾事務;
rollbackForClassname:指定一組異常類名字,其含義與<tx:method>中的rollback-for屬性語義完全一樣;
noRollbackFor:指定一組異常類,即使遇到該類異常也將提交事務,即不回滾事務;
noRollbackForClassname:指定一組異常類名字,其含義與<tx:method>中的no-rollback-for屬性語義完全一樣;
Spring提供的@Transaction註解事務管理內部同樣利用環繞通知TransactionInterceptor實現事務的開啟及關閉。
使用@Transactional註解事務管理需要特別注意以下幾點:
如果在介面、實現類或方法上都指定了@Transactional 註解,則優先順序順序為方法>實現類>介面;
建議只在實現類或實現類的方法上使用@Transactional,而不要在介面上使用,這是因為如果使用JDK代理機制是沒問題,因為其使用基於介面的代理;而使用使用CGLIB代理機制時就會遇到問題,因為其使用基於類的代理而不是介面,這是因為介面上的@Transactional註解是“不能繼承的”;
在Spring代理機制下(不管是JDK動態代理還是CGLIB代理),“自我呼叫”同樣不會應用相應的事務屬性,其語義和<tx:tags>中一樣;
預設只對RuntimeException異常回滾;
在使用Spring代理時,預設只有在public可見度的方法的@Transactional 註解才是有效的,其它可見度(protected、private、包可見)的方法上即使有@Transactional 註解也不會應用這些事務屬性的,Spring也不會報錯,如果你非要使用非公共方法註解事務管理的話,可考慮使用AspectJ。