事務原本是資料庫中的概念,在實際項目的開發中,進行事務的處理一般是在業務邏輯層, 即 Service 層。這樣做是為了能夠使用事務的特性來管理關聯操作的業務。 ...
1、Spring的事務管理:
- 事務原本是資料庫中的概念,在實際項目的開發中,進行事務的處理一般是在業務邏輯層, 即 Service 層。這樣做是為了能夠使用事務的特性來管理關聯操作的業務。
在 Spring 中通常可以通過以下兩種方式來實現對事務的管理:
(1)使用 Spring 的事務註解管理事務;使用 @Transactional 註解完成事務控制,此註解可添加到類上,則對類中所有方法執行事務的設定。
(2)使用 AspectJ 的 AOP 配置管理事務(聲明式事務管理);聲明式事務(必須掌握),在配置文件中添加一次,整個項目遵循事務的設定。
2、Spring中事務的五大隔離級別:
1)未提交讀(Read Uncommitted):允許臟讀,也就是可能讀取到其他會話中未提交事務修改的數據
2)提交讀(Read Committed):只能讀取到已經提交的數據。Oracle等多數資料庫預設都是該級別 (不重覆讀)
3)可重覆讀(Repeated Read):可重覆讀。在同一個事務內的查詢都是事務開始時刻一致的,InnoDB預設級別。在SQL標準中,該隔離級別消除了不可重覆讀,但是還存在幻象讀,但是innoDB解決了幻讀。
4)串列讀(Serializable):完全串列化的讀,每次讀都需要獲得表級共用鎖,讀寫相互都會阻塞。
5)使用資料庫預設的隔離級別 isolation = Isolation.DEFAULT
MySQL:mysql 預設的事務處理級別是'REPEATABLE-READ',也就是可重覆讀
Oracle:oracle 資料庫支持READ COMMITTED 和 SERIALIZABLE 這兩種事務隔離級別。預設系統事務隔離級別是READ COMMITTED,也就是讀已提交
3、Spring事務的傳播特性:
- 多個事務之間的合併,互斥等都可以通過設置事務的傳播特性來解決
常用:
PROPAGATION_REQUIRED:必被包含事務(增刪改必用)
PROPAGATION_REQUIRES_NEW:自己新開事務,不管之前是否有事務
PROPAGATION_SUPPORTS:支持事務,如果加入的方法有事務,則支持事務,如果沒有,不單開事務
PROPAGATION_NEVER:不能運行在事務中,如果包在事務中,拋異常
PROPAGATION_NOT_SUPPORTED:不支持事務,運行在非事務的環境
不常用:
PROPAGATION_MANDATORY:必須包在事務中,沒有事務則拋異常
PROPAGATION_NESTED:嵌套事務
4、基於註解的事務添加步驟:
1)在applicationContext_service.xml文件中添加事務管理器(事務管理器用來生成相應技術的連接+執行語句的對象.)
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!--因為事務必須關聯資料庫處理,所以要配置數據源-->
<property name="dataSource" ref="dataSource"></property>
</bean>
2)在applicationContext_service.xml文件中添加事務的註解驅動
<tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>
3)在業務邏輯的實現類上添加註解:@Transactional(propagation = Propagation.REQUIRED)
REQUIRED表示增刪改操作時必須添加的事務傳播特性
5、@Transactional註解參數詳解:
@Transactional(propagation = Propagation.REQUIRED,//事務的傳播特性
noRollbackForClassName = "ArithmeticException", //指定發生什麼異常不回滾(使用的是異常的名稱)
noRollbackFor = ArithmeticException.class,//指定發生什麼異常不回滾(使用的是異常的類型)
rollbackForClassName = "",//指定發生什麼異常必須回滾(使用的是異常的名稱)
rollbackFor = ArithmeticException.class,//指定發生什麼異常必須回滾(使用的是異常的類型)
timeout = -1, //連接超時設置,預設值是-1,表示永不超時
readOnly = false, //預設是false,如果是查詢操作,必須設置為true.
isolation = Isolation.DEFAULT//使用資料庫自已的隔離級別
)
6、聲明式事務:
Spring 的聲明式事務要求項目中的方法命名有規範
1)完成增加操作包含 add save insert set
2)更新操作包含 update change modify
3)刪除操作包含 delete drop remove clear
4)查詢操作包含 select find search get
配置事務切麵時可以使用通配符*來匹配所有方法。
<!--添加事務管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--配置事務切麵-->
<tx:advice id="myadvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*select*" read-only="true"/>
<tx:method name="*find*" read-only="true"/>
<tx:method name="*search*" read-only="true"/>
<tx:method name="*get*" read-only="true"/>
<tx:method name="*insert*" propagation="REQUIRED" no-rollback-for="ArithmeticException"/>
<tx:method name="*add*" propagation="REQUIRED"/>
<tx:method name="*save*" propagation="REQUIRED" no-rollback-for="ArithmeticException"/>
<tx:method name="*set*" propagation="REQUIRED"/>
<tx:method name="*update*" propagation="REQUIRED"/>
<tx:method name="*change*" propagation="REQUIRED"/>
<tx:method name="*modify*" propagation="REQUIRED"/>
<tx:method name="*delete*" propagation="REQUIRED"/>
<tx:method name="*remove*" propagation="REQUIRED"/>
<tx:method name="*drop*" propagation="REQUIRED"/>
<tx:method name="*clear*" propagation="REQUIRED"/>
<tx:method name="*" propagation="SUPPORTS"/>
</tx:attributes>
</tx:advice>
<!--綁定切麵和切入點-->
<aop:config>
<aop:pointcut id="mycut" expression="execution(* com.bjpowernode.service.impl.*.*(..))"></aop:pointcut>
<aop:advisor advice-ref="myadvice" pointcut-ref="mycut"></aop:advisor>
</aop:config>
情景:
在不同的事務傳播特性下,UsersServiceImpl類中方法中調用AccountServiceImpl類中的方法(此訪問中拋出異常),推測最後資料庫中users表和accounts是否成功更新了數據。