`
如若_晴
  • 浏览: 110458 次
  • 性别: Icon_minigender_1
  • 来自: 济南
社区版块
存档分类
最新评论

Spring事务管理

 
阅读更多
主要介绍了Spring事务管理的原理和基于XML的事务管理方式
1. Spring事务管理简介
(1)Spring为多种不同类型的事务管理机制提供统一编程模型,这些事务管理模型包括JTA、JDBC、Hibernate、JPA和JDO。
(2)Spring支持声明式事务管理(使用XML文档配置(或者Annotation)结合AOP实现的事务管理)。
(3)为代码嵌入式(programmatic)的事务管理提供API接口,与复杂的JTA接口相比要简单的多。
(4)能够与Spring的数据抽象访问完美结合。
2. Spring事务管理的优点
一般认为,JavaEE中有两种类型的事务管理方式,分别是全局(global)和局部(local)事务。它们各自有优缺点。
2.1 全局事务
全局事务主要是指通过JTA管理多个数据库或者消息队列的事务处理。JTA接口使用复杂,并且必须结合JNDI才能使用,因此限制了应用代码潜在的重用性。
此外,EJB的CMT(Container Manager Transaction),使用声明式事务管理,无需使用JNDI的支持,但是需要引入EJB相应的服务。
2.2 局部事务
局部事务是资源(数据库)相关的,例如JDBC的connection。尽管局部事务使用简单,但是其缺点也是很明显的:局部事务不能在多个源(数据库)间统一管理事务,此外事务管理使用嵌入式代码,不利于代码逻辑的清晰展现。
2.3 Spring一致性编程模型
Spring解决了全局和局部事务的缺陷,使用一致性编程模型,一次编码,可以在不同的事务策略和运行环境中迁移。此外,Spring提供两种类型的事务管理方式,分别是代码嵌入式和声明式,更多的人喜欢使用声明式。

笔者注:Spring的事务管理依赖于底层的具体事务管理框架,比如,如果底层的事务管理框架是全局的(例如JTA、或者是CMT),那么Spring的事务管理就是全局的,如果底层的事务管理是局部的(例如:Hibernate、JDBC,JDO等),则Spring的事务管理就是局部的。在很多情况下,即使使用的是EJB的CMT管理多个数据源(数据库),可能仍然会选择使用Spring事务管理应用,因为Spring提供一种声明式的事务管理机制,在对已有代码进行零改动(使用XML+AOP)或者略微改动(使用Annotation+AOP)的情况下,统一管理事务,因此,有利于代码在不同的事务策略和运行环境中迁移。
3. Spring事务抽象
Spring的事务管理通过org.springframework.transaction.PlatformactionManager接口表示:
public interface PlatformTransactionManager {

  TransactionStatus getTransaction(TransactionDefinition definition)
    throws TransactionException;

  void commit(TransactionStatus status) throws TransactionException;

  void rollback(TransactionStatus status) throws TransactionException;
}
PlatformTransactionManager是一个服务提供商接口(SPI),针对不同类型的底层事务框架,Spring提供了不同的PlatFormTransactionManager实现版本,在使用过程中选择适当的实现版本管理底层事务框架即可,常见的有:
(1)JDBC org.springframework.jdbc.datasource.DataSourceTransactionManager
(2)JTA  org.springframework.transaction.jta.JtaTransactionManager
(3)Hibernate org.springframework.orm.hibernate3.HibernateTransactionManager
下面分析PlatFormTransactionManager中的方法,getTransaction方法,通过TransactionDefinition传入参数返回TransactionStatus对象。TransactionStatus可能表示一个新的事务,或者表示一个已经存在的事务(如果当前线程的调用栈中已经存在一个事务)。在JavaEE中经常将事务与一个具体的执行线程相关联,因此,在获取事务时,如果当前线程中已经存在事务,即将存在的事务返回,否则创建一个新事务返回。
TransactionDefinition接口规范如下:
(1)Isolation:事务隔离等级(是数据库事务的一个概念,不同数据库支持不同的事务隔离等级)
(2)Propagation:事务传播方式,通过设置可以决定,当一个事务方法准备执行前,线程栈中已经存在一个事务,可选的策略为:使用先前的事务作为即将执行的方法事务,或者,将先前的事务暂停,创建一个全新的事务处理当前方法,处理结束后,在返回先前的事务。
(3)Timeout:如果事务持续的时间超时,则启动底层事务管理的回滚机制,回滚当前事务。
(4)Read-only status:Read-only事务主要应用于代码中仅读取数据,并未对数据修改的情况。在某些情况下,Read-only事务可以优化事务管理,例如底层事务框架使用Hibernate时。
TransactionStatus提供了简单的方式控制事务执行和查询事务状态,如下所示:
public interface TransactionStatus extends SavepointManager {

    boolean isNewTransaction();

    boolean hasSavepoint();

    void setRollbackOnly();

    boolean isRollbackOnly();

    void flush();

    boolean isCompleted();

}
针对不同的底层事务框架,需要选择正确的PlatformTransactionManager实现版本,下面提供Hibernate的配置方法:
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
  <property name="dataSource" ref="dataSource" />
  <property name="mappingResources">
  <list>
    <value>org/springframework/samples/petclinic/hibernate/petclinic.hbm.xml</value>
  </list>
  </property>
  <property name="hibernateProperties">
    <value>
      hibernate.dialect=${hibernate.dialect}
    </value>
  </property>
</bean>

<bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
  <property name="sessionFactory" ref="sessionFactory" />
</bean>
从上述配置文件可以看出,需要在txManager中指定Hibernate Sessionfactory。在Hibernate中通过Sessionfactory中的OpenSession方法创建Session,之后使用Session中的beginTransaction方法开启事务。将sessionFactory注入到HibernateTransactionManager中的属性后,Spring即可通过这个sessionFactory控制Hibernate事务管理。所以,Spring事务管理仍然是依赖于底层的事务管理框架,而Spring只是提供对事务的一种抽象,这种抽象能够在多种事务框架间迁移。当需要新的事务管理框架和策略支持时,无需更改代码,只需更改配置文件即可完成这种事务框架间的迁移。
4. 使用事务同步资源(略)
5.声明式(Declarative)事务管理
声明式事务管理以EJB的CMT十分类似,它们的主要区别在于:
(1)CMT依赖于JTA,只适用于全局事务管理,Spring不单可以适用于JTA,同时也适用于局部事务管理(例如:JDBC,JPA,Hibernate和JDO等)。
(2)Spring声明式事务管理可以应用于任意的类管理,而EJBs只是用于特殊类管理。
(3)Spring的回滚规则与EJB不同。
(4)Spring可以通过AOP规则在回滚过程中插入用户定义行为,也可以同事务管理advice一起定义任意的advice。(advice是AOP的概念)
(5)Spring不支持跨越多个远程调用的事务管理。一般认为Spring的事务最多在一个请求范围内完成,如果需要跨越多个请求过程管理事务,那么应当选择EJB,但是这种情况并不多见。
回滚规则的制定十分重要。当然,可以使用传统的方式,调用TransactionStatus中的setRollback方法,通过代码回滚事务;但是,更常用的是定义回滚规则,当应用程序执行过程中满足了某项回滚规则(定义制定Runtime exception出发事务回滚),当前事务应当自动回滚。通过Spring的声明式事务管理,无需在代码中加入任何与事务相关的内容,即可完成按回滚规则管理事务的方式。Spring与EJB相似,仅对未处理的运行时异常(runtime exception)回滚。
5.1 理解声明式事务实现机制
Spring通过元数据(metadata,包括XML或者annotation)配置声明式事务管理,并使用AOP代理机制最终实现事务管理。过程大致是,开发者通过元数据标明需要事务管理的方法(通过AOP中的cut-point),同时提供事务管理的策略advice(通过TX的advice),并结合实际提供的PlatformTransactionManager。当用户调用了事务管理方法时,系统使用具体的PlatformTransactionManager实现,根据advice提供的策略创建(或者获取)一个事务,之后执行方法内部逻辑,当出现异常时,回滚当前事务,否则执行结束,提交事务,清理当前事务。整个过程是使用AOP代理完成的,AOP代理在方法代码前添加事务启动逻辑,在方法执行后添加事务提交和清理逻辑,并且代理监控方法中可能出现的异常,当有未捕获异常抛出时,代理使用相应的回滚逻辑回滚事务。(有关代理的更进一步说明请参考Spring官方文档AOP章节)。整个过程如下图所示:
5.2 声明式事务管理举例
// the service interface that we want to make transactional

package x.y.service;

public interface FooService {

  Foo getFoo(String fooName);

  Foo getFoo(String fooName, String barName);

  void insertFoo(Foo foo);

  void updateFoo(Foo foo);

}
// an implementation of the above interface

package x.y.service;

public class DefaultFooService implements FooService {

  public Foo getFoo(String fooName) {
    throw new UnsupportedOperationException();
  }

  public Foo getFoo(String fooName, String barName) {
    throw new UnsupportedOperationException();
  }

  public void insertFoo(Foo foo) {
    throw new UnsupportedOperationException();
  }

  public void updateFoo(Foo foo) {
    throw new UnsupportedOperationException();
  }

}
下面声明get方法为read-only,而其他方法为默认。
<bean id="fooService" class="x.y.service.DefaultFooService"/>

  <!-- the transactional advice (what 'happens'; see the <aop:advisor/> bean below) -->
  <tx:advice id="txAdvice" transaction-manager="txManager">
  <!-- the transactional semantics... -->
  <tx:attributes>
    <!-- all methods starting with 'get' are read-only -->
    <tx:method name="get*" read-only="true"/>
    <!-- other methods use the default transaction settings (see below) -->
    <tx:method name="*"/>
  </tx:attributes>
  </tx:advice>

  <!-- ensure that the above transactional advice runs for any execution
    of an operation defined by the FooService interface -->
  <aop:config>
  <aop:pointcut id="fooServiceOperation" expression="execution(* x.y.service.FooService.*(..))"/>
  <aop:advisor advice-ref="txAdvice" pointcut-ref="fooServiceOperation"/>
  </aop:config>

  <!-- don't forget the DataSource -->
  <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
  <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver"/>
  <property name="url" value="jdbc:oracle:thin:@rj-t42:1521:elvis"/>
  <property name="username" value="scott"/>
  <property name="password" value="tiger"/>
  </bean>

  <!-- similarly, don't forget the PlatformTransactionManager -->
  <bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  <property name="dataSource" ref="dataSource"/>
  </bean>

正如上例所述的那样,通过<tx:advice/>标签设置了“所有以get开始的方法需要使用read-only事务管理策略,而其它方法使用默认的事务管理策略”。transaction-manager属性用于指定具体的PlatformTransactionManager实现,txAdvice需要使用txManager驱动具体的事务管理过程。因此,pointcut标明需要添加事务代理的方法,通过advice配置了事务属性和事务管理者,当需要事务支持的方法被调用时,aop使用advice找到PlatformTransactionManager,而PlatformTransactionManager使用底层的具体事务管理架构,最终实现事务管理功能。
5.3 回滚配置
默认情况下,Spring只会对方法中的未检测运行异常回滚,而被检测的异常不会导致事务回滚。
可以通过如下配置方法甚至配置一个检测的异常回滚:
<tx:advice id="txAdvice" transaction-manager="txManager">
  <tx:attributes>
  <tx:method name="get*" read-only="true" rollback-for="NoProductInStockException"/>
  <tx:method name="*"/>
  </tx:attributes>
</tx:advice>
当然也可以使用下面方法,使一个未检测异常不发生回滚,当这个异常发生时,Spring将提交事务(而不是回滚):
<tx:advice id="txAdvice">
  <tx:attributes>
  <tx:method name="updateStock" no-rollback-for="InstrumentNotFoundException"/>
  <tx:method name="*"/>
  </tx:attributes>
</tx:advice>
5.4为不同的bean配置不同的advice
<aop:config>

    <aop:pointcut id="defaultServiceOperation"
          expression="execution(* x.y.service.*Service.*(..))"/>

    <aop:pointcut id="noTxServiceOperation"
          expression="execution(* x.y.service.ddl.DefaultDdlManager.*(..))"/>

    <aop:advisor pointcut-ref="defaultServiceOperation" advice-ref="defaultTxAdvice"/>

    <aop:advisor pointcut-ref="noTxServiceOperation" advice-ref="noTxAdvice"/>

  </aop:config>

  <!-- this bean will be transactional (see the 'defaultServiceOperation' pointcut) -->
  <bean id="fooService" class="x.y.service.DefaultFooService"/>

  <!-- this bean will also be transactional, but with totally different transactional settings -->
  <bean id="anotherFooService" class="x.y.service.ddl.DefaultDdlManager"/>

  <tx:advice id="defaultTxAdvice">
    <tx:attributes>
      <tx:method name="get*" read-only="true"/>
      <tx:method name="*"/>
    </tx:attributes>
  </tx:advice>

  <tx:advice id="noTxAdvice">
    <tx:attributes>
      <tx:method name="*" propagation="NEVER"/>
    </tx:attributes>
  </tx:advice>
分享到:
评论

相关推荐

    Spring事务管理Demo

    Spring事务管理的目的是确保数据的一致性和完整性,尤其是在多操作、多资源的环境中。本Demo将深入探讨Spring如何实现事务的管理。 首先,Spring提供了两种主要的事务管理方式:编程式事务管理和声明式事务管理。 ...

    spring事务管理

    ### Spring事务管理详解 #### 一、事务管理基础概念 在深入探讨Spring事务管理之前,我们需要先理解什么是事务。事务可以被定义为一系列的操作集合,这些操作作为一个整体被提交或回滚。简单来说,事务就是一个不...

    Spring事务管理开发必备jar包

    本资源包提供了进行Spring事务管理开发所需的所有关键库,包括框架基础、核心组件、AOP(面向切面编程)支持、日志处理、编译工具以及与数据库交互的相关jar包。下面将对这些知识点进行详细解释: 1. **Spring框架*...

    spring 事务管理的理解

    Spring 框架是Java开发中...理解并熟练掌握Spring事务管理,对于提升应用程序的稳定性和可靠性至关重要。在实际开发中,结合声明式事务管理、事务传播行为、隔离级别和回滚规则,可以有效地确保数据的完整性和一致性。

    Spring事务管理.pdf

    Spring事务管理.pdf 1.资料 2.本地事务与分布式事务 3.编程式模型 4.宣告式模型

    Synchronized锁在Spring事务管理下线程不安全

    Synchronized锁在Spring事务管理下,导致线程不安全。

    Spring事务管理的jar包

    本篇将深入探讨Spring事务管理的核心概念、工作原理以及如何使用`spring-tx-3.2.0.RELEASE.jar`这个jar包。 首先,我们需要理解什么是事务。在数据库系统中,事务是一组操作,这些操作被视为一个整体,要么全部完成...

    Spring事务管理失效原因汇总

    标题“Spring事务管理失效原因汇总”指出了本文的核心内容是分析在使用Spring框架进行事务管理时可能遇到的问题及其原因。描述部分进一步说明了事务失效的后果往往不明显,容易在测试环节被忽略,但在生产环境中出现...

    spring事务管理5种方法

    本篇文章将深入探讨Spring事务管理的五种方法,旨在帮助开发者更好地理解和运用这一核心特性。 首先,我们来了解什么是事务。在数据库操作中,事务是一组逻辑操作,这些操作要么全部成功,要么全部失败,确保数据的...

    Spring事务管理4种方式

    本文将详细介绍Spring事务管理的四种方式:编程式事务管理、声明式事务管理、PlatformTransactionManager接口以及TransactionTemplate。 1. **编程式事务管理**:这是一种手动控制事务的方式,通过在代码中调用`...

    详细介绍Spring事务管理

    ### Spring事务管理详解 #### 一、Spring事务管理的重要性及必要性 在现代软件开发中,事务管理是一项至关重要的技术,特别是在涉及数据库操作时。事务能够确保一系列操作要么全部成功,要么全部失败,这对于保持...

    spring事务管理.rar

    Spring事务管理是Spring框架的核心特性之一,它提供了一种强大且灵活的方式来管理应用程序中的事务边界。在企业级Java应用中,事务处理是确保数据一致性、完整性和可靠性的关键部分。本篇文章将深入探讨Spring的事务...

    spring事务管理.doc

    总的来说,Spring事务管理提供了一种灵活、强大的方式来处理应用程序中的事务,无论是在简单还是复杂的事务场景下,都能有效保证数据的一致性和完整性。通过声明式事务管理,开发者可以将关注点从事务细节中解脱出来...

    Spring事务管理配置文件问题排查

    当出现像描述中那样的问题——SQL语句执行出错但事务未回滚时,我们需要深入理解Spring事务管理的配置和机制。以下是一些关键知识点: 1. **Spring事务管理类型**: - **编程式事务管理**:通过`...

    spring事务管理几种方式代码实例

    spring事务管理几种方式代码实例:涉及编程式事务,声明式事务之拦截器代理方式、AOP切面通知方式、AspectJ注解方式,通过不同方式实例代码展现,总结spring事务管理的一般规律,从宏观上加深理解spring事务管理特性...

    Spring事务管理和SpringJDBC思维导图

    在思维导图"Spring Transaction.twd"中,可能包含了Spring事务管理的各个概念和它们之间的关系,如事务的ACID属性(原子性、一致性、隔离性和持久性),事务管理器,以及声明式和编程式事务管理的实现方式。...

Global site tag (gtag.js) - Google Analytics