【SSM框架】Spring筆記 — 事務詳解

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是否成功更新了數據。