Spring事务的传播特性和隔离级别

Spring 对事务控制的支持统一在 TransactionDefinition 类中描述,该类有以下几个重要的接口方法:

  • int getPropagationBehavior():事务的传播行为

  • int getIsolationLevel():事务的隔离级别

  • int getTimeout():事务的过期时间

  • boolean isReadOnly():事务的读写特性。

事务的7种传播特性

所谓事务传播行为就是多个事务方法相互调用时,事务如何在这些方法间传播。Spring 支持 7 种事务传播行为:

  • 1、PROPAGATION_REQUIRED(默认):如果存在一个事务,则支持当前事务。如果没有事务则开启

  • 2、PROPAGATION_SUPPORTS:如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行

  • 3、PROPAGATION_MANDATORY:如果已经存在一个事务,支持当前事务。如果没有一个活动的事务,则抛出异常。

  • 4、PROPAGATION_REQUIRES_NEW:总是开启一个新的事务。如果一个事务已经存在,则将这个存在的事务挂起。

  • 5、PROPAGATION_NOT_SUPPORTED:总是非事务地执行,并挂起任何存在的事务。

  • 6、PROPAGATION_NEVER:总是非事务地执行,如果存在一个活动事务,则抛出异常

  • 7、PROPAGATION_NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中. 如果没有活动事务,  则按TransactionDefinition.PROPAGATION_REQUIRED 属性执行

Spring事务的传播特性和隔离级别

事物传递思维导图

Spring事务的传播特性和隔离级别


Spring事务的隔离级别

  • 1、ISOLATION_DEFAULT:这是一个PlatfromTransactionManager默认的隔离级别,使用数据库默认的事务隔离级别,另外四个与JDBC的隔离级别相对应

  • 2、ISOLATION_READ_UNCOMMITTED:这是事务最低的隔离级别,它充许令外一个事务可以看到这个事务未提交的数据。这种隔离级别会产生脏读,不可重复读和幻像读。

  • 3、ISOLATION_READ_COMMITTED:保证一个事务修改的数据提交后才能被另外一个事务读取。另外一个事务不能读取该事务未提交的数据

  • 4、ISOLATION_REPEATABLE_READ:这种事务隔离级别可以防止脏读,不可重复读。但是可能出现幻像读。它除了保证一个事务不能读取另一个事务未提交的数据外,还保证了避免下面的情况产生(不可重复读)。

  • 5、ISOLATION_SERIALIZABLE:这是花费最高代价但是最可靠的事务隔离级别。事务被处理为顺序执行。除了防止脏读,不可重复读外,还避免了幻像读。 

Spring事务的传播特性和隔离级别

脏读:指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。因为这个数据是还没有提交的数据, 那么另外一 个事务读到的这个数据是脏数据,依据脏数据所做的操作可能是不正确的。

不可重复读:指在一个事务内,多次读同一数据。在这个事务还没有结束时,另外一个事务也访问该同一数据。 那么,在第一个事务中的两次读数据之间,由于第二个事务的修改,那么第一个事务两次读到的数据可能是不一样的。这样就发生了在一个事务内两次读到的数据是不一样的,因此称为是不可重复读。

幻觉读:指当事务不是独立执行时发生的一种现象,例如第一个事务对一个表中的数据进行了修改,这种修改涉及 到表中的全部数据行。同时,第二个事务也修改这个表中的数据,这种修改是向表中插入一行新数据。那么,以后就会发生操作第一个事务的用户发现表中还有没有修改的数据行,就好象发生了幻觉一样。

总结:

隔离级别越高,越能保证数据的完整性和一致性,但是对并发性能的影响也越大。

大多数的数据库默认隔离级别为 Read Commited,比如 SqlServer、Oracle

少数数据库默认隔离级别为:Repeatable Read 比如: MySQL InnoDB

详解参考文章:【Oracle】数据库事务隔离级别

Spring与Hibernate的集成,使用HibernateDaoSupport的配置示例

在ssh框架应用中,Spring与Hibernate的事务集成基本上是比较固定的,我们把事务的集成单独配置到applicationContext-common.xml中:

<?xml version="1.0"encoding="UTF-8"?> 
<beansxmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
       http://www.springframework.org/schema/aop
       http://www.springframework.org/schema/aop/spring-aop-2.0.xsd
       http://www.springframework.org/schema/tx
       http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
           
   <!--配置SessionFactory -->
   <beanid="sessionFactory"class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
      <propertyname="configLocation">
         <value>classpath:hibernate.cfg.xml</value>
      </property>
   </bean>
  
   <!--配置事务管理器 -->
   <beanid="transactionManager"class="org.springframework.orm.hibernate3.HibernateTransactionManager">
      <propertyname="sessionFactory">
         <refbean="sessionFactory"/>        
      </property>
   </bean>
  
   <!--那些类那些方法使用事务 -->
   <aop:config>
      <aop:pointcutid="allManagerMethod" expression="execution(*com.bjpowernode.usermgr.manager.*.*(..))"/>
      <aop:advisorpointcut-ref="allManagerMethod" advice-ref="txAdvice"/>
   </aop:config>
  
   <!--事务的传播特性 --> 
   <tx:adviceid="txAdvice" transaction-manager="transactionManager">
      <tx:attributes>
         <tx:methodname="add*" propagation="REQUIRED"/>
         <tx:methodname="del*" propagation="REQUIRED"/>
         <tx:methodname="modify*" propagation="REQUIRED"/>
         <tx:methodname="*" propagation="REQUIRED" read-only="true"/>
      </tx:attributes>
   </tx:advice>
</beans>


未经允许请勿转载:程序喵 » Spring事务的传播特性和隔离级别

点  赞 (3) 打  赏
分享到: