转载

Spring的事务管理机制

在我的博客(http://www.hollischuang.com/)中,多篇文章介绍了事务相关的内容,其中包括数据库的事务的相关介绍、分布式事务的相关介绍以及在 Spring中使用注解进行事务的配置方式 等。

本文在以上文章的基础上,一起来学习一下Spring中对事务的支持——Spring的事务管理机制。

Spring对事务管理的支持

与EJB类似,Spring提供了对 编码式声明式 事务管理的支持。但是,Spring对事务管理的能力远远超过EJB。这里就不详细介绍编码式事务和声明式事务的区别了。有兴趣的读者可以自行Google。

Spring对事务管理是通过事务管理器来实现的。Spring提供了许多内置事务管理器实现:

事务管理器

( org.springframework.* )

使用场景
jdbc.datasource.DataSourceTransactionManager 数据源事务管理器,提供对单个 javax.sql.DataSource 事务管理,用于Spring JDBC抽象框架、iBATIS或MyBatis框架的事务管理;
orm.jdo.JdoTransactionManager 提供对单个 javax.jdo.PersistenceManagerFactor y事务管理,用于集成JDO框架时的事务管理;
orm.jpa.JpaTransactionManager 提供对单个 javax.persistence.EntityManagerFactory 事务支持,用于集成JPA实现框架时的事务管理;
orm.hibernate3.HibernateTransactionManager 提供对单个 org.hibernate.SessionFactory 事务支持,用于集成Hibernate框架时的事务管理;该事务管理器只支持Hibernate3+版本,且Spring3.0+版本只支持Hibernate 3.2+版本;
transaction.jta.JtaTransactionManager 提供对分布式事务管理的支持,并将事务管理委托给Java EE应用服务器事务管理器;
transaction.jta.OC4JjtaTransactionManager Spring提供的对OC4J10.1.3+应用服务器事务管理器的适配器,此适配器用于对应用服务器提供的高级事务的支持;
transaction.jta.WebSphereUowTransactionManager Spring提供的对WebSphere 6.0+应用服务器事务管理器的适配器,此适配器用于对应用服务器提供的高级事务的支持;
transaction.jta.WebLogicJtaTransactionManager Spring提供的对WebLogic 8.1+应用服务器事务管理器的适配器,此适配器用于对应用服务器提供的高级事务的支持。

Spring的事务管理机制

以上就是Spring中支持使用的事务管理器,一般我们比较常用的就是 HibernateTransactionManagerDataSourceTransactionManager 。我们在使用事务的时候要声明要使用哪种事务管理器。如:

<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">     <property name="dataSource" ref="dataSource" /> </bean> 

事务属性

spring中,声明事务是通过事务属性来定义的。事务属性描述了事务策略如何应用到方法上事务属性包含5个方面:

传播行为

隔离级别

回滚规则

事务超时

是否只读

Spring的事务管理机制

这里简单介绍一下这五个属性。

传播行为

传播行为定义了客户端与被调用方法之间的事务边界,即传播规则回答了这样的一个问题,新的事务应该被启动还是挂起,或者方法是否要在事务环境中运行。(后面会有单独的文章介绍该属性)

隔离级别

隔离级别定义了一个事务可能受其他并发事务影响的程度。多事务并发可能会导致脏读、幻读、不可重复读等各种读现象。(具体参考:数据库的读现象浅析)

ISOLATION_DEFAULT:使用后端数据库默认的规则

ISOLATION_READ_UNCOMMITTED:允许读取尚未提交的数据变更,可能会导致脏读,幻读或不可重复读

ISOLATION_READ_COMMITTED:允许读取并发事务已经提交的数据,可以防止脏读,但是幻读或不可重复读仍有可能发生

ISOLATION_REPEATABLE_READ:对同意字段的多次读取结果是一致的,除非数据是被本事务自己所修改,看阻止脏读和不可重复读,但幻读仍有可能发生

ISOLATIOM_SERIALIZABLE:完全服从ACID的隔离级别,确保阻止脏读,不可重复读以及幻读,这是最慢的数据隔离级别

(具体参考: 深入分析事务的隔离级别 )

是否只读

如果事务只对后端的数据库进行读操作,数据库可以利用事务ID只读特性来进行一些特定的优化。通过将事务设置为只读,你就可以给数据库一个机会,让他应用它认为合适的优化措施。因为是否只读是在事务启动的时候由数据库实施的,所以只有对那些具备可能启动一个新事务的传播行为( PROPAGATION_REQUIRED , PROPAGATION_REQUIRED_NEW , PROPAGATION_NESTED )的方法来说,才有意义。

事务超时

为了使应用程序很好地运行,事务不能运行太长时间。因为超时时钟会在事务开始时启动,所以只有对那些具备可能启动一个新事务的传播行为( PROPAGATION_REQUIRED , PROPAGATION_REQUIRED_NEW , PROPAGATION_NESTED )的方法来说,才有意义。

事务回滚

事务回滚规则定义了哪些异常会导致事务回滚而哪些不会。默认情况下,事务只有在遇到运行时期异常才回滚,而在遇到检查型异常时不会回滚。

配置方式

事务属性的配置方式通过以下关键字来指定:

关键字 含义
isolation 指定事务的隔离级别
propagation 定义事务的传播规则
read-only 指定事务为只读
rollback-forno-rollback-for rollback-for指定事务对哪些检查型异常应当回滚而不提交no-rollback-for指定事务对哪些异常应当继续执行而不回滚
timeout 对于长时间运行的事务定义超时时间

XML中事务属性的配置方式如下:

<tx:advice id="txAdvice" transactionmanager="transactionManager">         <tx:attributes>             <tx:method name="*" propagation="REQUIRED" read-only="true"/>         </tx:attributes> </tx:advice> 

注解中事务属性的配置方式如下:

@Transaction(propagation=Propagation. REQUIRED,readOnly=true) public void add(String username){     //... } 

总结

事务是企业应用开发中很重要的组成部分,他让软件变得更加健壮。他保证了全有或全无的操作。

Spring同时支持编码式和声明式事务管理,无论使用哪种方式进行事务管理,都应该知道与事务相关的五个属性。

参考资料

《Spring实战》

spring 事务属性

(全文完)

原文  http://www.hollischuang.com/archives/1489
正文到此结束
Loading...