事務原本是資料庫中的概念,在實際項目的開發中,進行事務的處理一般是在業務邏輯層, 即 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是否成功更新了資料。
