关于spring中事务管理的几件小事

1.Spring中的事务管理

  • 作为企业级应用程序框架,Spring在不同的事务管理API之上定义了一个抽象层。而应用程序开发人员不必了解底层的事务管理API,就可以使用Spring的事务管理机制。
  • Spring既支持编程式事务管理,也支持声明式的事务管理。
  • 编程事务管理:将事务管理代码嵌入到业务方法中来控制事务的提交和回滚。在编程式管理事务时,必须在每个事务操作中包含额外的事务管理代码。
  • 声明式事务管理:大多数情况下比编程式事务管理更好用。它将事务管理代码从业务方法中分离出来,以声明的方式来实现事务管理。事务管理作为一种横切关注点,可以通过AOP方法模块化。Spring通过SpringAOP框架支持声明式事务管理。

2.Spring中的事务管理器

  • Spring从不同的事务管理API中抽象了一整套的事务机制。开发人员不必了解底层的事务API,就可以利用这些事务机制。有了这些事务机制,事务管理代码就能独立于特定的事务技术了。
  • Spring的核心事务管理抽象是TransactionManager管理封装了一组独立于技术的方法。无论使用Spring的那种事务管理策略(编程式或声明式),事务管理器都是必须的。
  • DataSourceTransactionManager:在应用程序中只需要处理一个数据源,而且通过JDBC存储。
  • JtaTransactionManager:在JavaEES应用服务器上用JTA进行事务管理。
  • HibernateTransactionManager:用Hibernate框架存取数据库。
  • 事务管理器以普通的Bean形式声明在SpringIOC容器中。

3.用事务通知声明式地管理事务

  • 事务管理是一种横切关注点。
  • 为了在Spring 2.x中启用声明式事务管理,可以通过tx Schema中定义的tx:advice元素声明事务通知。为此必须事先将这个Schema定义添加到根元素中去。
  • 声明了事务通知后,就需要将它与切入点关联起来,由于事务通知是在aop:config元素外部声明的,索引它无法直接与切入点产生关联,所以必须在aop:config元素中声明一个增强器通知与切入点关联起来。
  • 由于Spring AOP是基于代理的方法,索引只能增强公共方法。因此,只有公有方法才能通过Spring AOP进行事务管理。
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
         <property name="dataSource" ref="dataSource"></property>
    </bean>
    
    <tx:advice id="bookShopTxAdvice" transaction-manager="transactionManager"></tx:advice>
    
    <aop:config>
        <aop:pointcut id="bookShopOperation" expression="execution(* *.BookShopService.*(..))"></aop:pointcut>
        <aop:advisor advice-ref="bookShopTxAdvice" pointcut-ref="bookShopOperation"/>
    </aop:config>

4.用@Transactional注解声明式的管理事务

  • 除了在带有切入点,通知和增强器的Bean配置文件中声明事务外,Spring还允许简单地用@Transactional注解来标注事务方法,。
  • 为了将方法定义为支持事务处理的,可以为方法添加@Transactional注解。根据Spring AOP基于代理机制,只能标注公有方法。
  • 可以在方法或者类级别上添加@Transactional注解。当把这个注解应用到类上时,这个类中的所有公共方法都会被定义成支持事务处理的。
  • 在Bean配置文件中只需要启用tx:annotation-driven元素,并为之指定事务处理器就可以了。
  • 如果事务处理器的名称是transactionManager,就可以在tx:annotation-driven元素中省略transaction-manager属性。这个元素会自动检测该名称的事务处理器。
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    
    <context:component-scan base-package="com.desparado.service"></context:component-scan>
    
    <tx:annotation-driven/>

5.事务传播属性

  • 当事务方法被另一个事务方法调用时,必须指定事务应该如何传播,例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。
  • 事务的传播行为可以由传输属性指定,Spring定义了7种传播行为。
传播属性 描述
REQUIRED 如果有事务在运行,当前的方法就在这个事务内部运行,否则,就启动一个新的事务,并在自己的事务内运行。
REQUIRED_NEW 当前的方法必须启动新事务,并在它自己的事务内运行,如果有事务正在运行,应该将它挂起
SUPPORTS 如果有事务在运行,当前的方法就在这个事务内运行,否则它可以不允许在事务中
NOT_SUPPORTED 当前的方法不应该运行在事务中,如果有运行的事务,将它挂起
MANDATORY 当期的方法必须运行在事务内部,如果没没有正在运行的事务,就抛出异常
NEVER 当前的方法不应该运行在事务中,如果有运行的事务,就跑出异常
NESTED 如果有事务在运行,当前的方法就应该在这个事务的嵌套事务内运行,否则,就启动一个新的事务,并在它自己的事务内运行
  • 事务传播属性的配置
    在@Transactional注解中的属性中进行配置
@Transactional(propagation=Propagation.REQUIRES_NEW)

在xml中可以在tx:method中进行配置

<tx:advice id="bookShopTxAdvice" transaction-manager="transactionManager">
    <tx:attributes>
          <tx:method name="purchase" propagation="REQUIRES_NEW"/>
    </tx:attributes>
</tx:advice>
    

6.设置隔离事务属性

  • 用@Transactional注解声明式地管理事务时可以在@Transactional的isolation属性中设置隔离级别
@Transactional(propagation=Propagation.REQUIRES_NEW,isolation=Isolation.READ_COMMITTED)

  • 在Spring 2.x事务通知中,可以在<tx:method》元素中指定隔离级别
<tx:advice id="bookShopTxAdvice" transaction-manager="transactionManager">
    <tx:attributes>
          <tx:method name="purchase" propagation="REQUIRES_NEW" 
              isolation="READ_COMMITTED"/>
    </tx:attributes>
</tx:advice>

7.设置回滚事务属性

  • 默认情况下只有未检查异常(RuntimeException和Error类型的异常)会导致事务回滚,二受检查异常不会。
  • 事务的回滚规则可以通过@Transactional注解的rollbackFor和noRollbackFor属性来定义,这两个属性被声明为Class[]类型的,因此可以为这两个属性指定多个异常类。
    - rollbackFor:遇到时必须进行回滚。
    - noRollbackFor:一组异常类,遇到时必须不会滚
@Transactional(propagation=Propagation.REQUIRED_NEW,
                isolation=Isolation.READ_COMMITTED,
                rollbackFor=(IOException.class,SQLException.class),
                noRollbackFor=ArithmeticException.class)  
public void purchase(String isbn,String username){}
  • 在Spring 2.x事务通知中,可以在<tx:method》原始种指定回滚规则,如果有不止一种异常,用逗号隔开
<tx:advice id="bookShopTxAdvice" transaction-manager="transactionManager">
   <tx:attributes>
         <tx:method name="purchase" propagation="REQUIRES_NEW" 
             isolation="READ_COMMITTED"
             rollback-for="java.io.IOException,java.sql.SQLException" 
             no-rollback-for="java.lang,ArithmeticException"/>
   </tx:attributes>
</tx:advice>

8.超时和只读属性

  • 由于事务可以在行和表上获得锁,因此长事务会占用资源,并对整体性能产生影响。
  • 如果一个事务只读取数据但不做修改,数据库引擎可以对这个事务进行优化。
  • 超时事务属性:事务在强制回滚之前可以保持多久。这样可以防止长期运行的事务占用资源。
  • 只读事务属性:表示这个事务只读取数据库但不更新数据库,这样可以帮助数据库引擎优化事务。
  • 超时和只读属性可以在@Transactional注解中定义,超时属性以秒为单位。
@Transactional(propagation=Propagation.REQUIRED_NEW,
                isolation=Isolation.READ_COMMITTED,
                rollbackFor=(IOException.class,SQLException.class),
                noRollbackFor=ArithmeticException.class,
                readOnly=true,tomeout=30)  
public void purchase(String isbn,String username){}

-在Spring 2.x事务通知中,超时和只读属性可以在tx:method元素中进行指定

<tx:advice id="bookShopTxAdvice" transaction-manager="transactionManager">
    <tx:attributes>
          <tx:method name="purchase" propagation="REQUIRES_NEW" 
              isolation="READ_COMMITTED"
              rollback-for="java.io.IOException,java.sql.SQLException" 
              no-rollback-for="java.lang,ArithmeticException"  
              read-only=true
              timeout=30/>
    </tx:attributes>
</tx:advice>
posted @ 2019-05-31 02:07  一条路上的咸鱼  阅读(223)  评论(0编辑  收藏  举报