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

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

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

5.事务传播属性

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

6.设置隔离事务属性

  • 用@Transactional注解声明式地管理事务时可以在@Transactional的isolation属性中设置隔离级别
1
@Transactional(propagation=Propagation.REQUIRES_NEW,isolation=Isolation.READ_COMMITTED)
Copied!
  • 在Spring 2.x事务通知中,可以在<tx:method》元素中指定隔离级别
1
<tx:advice id="bookShopTxAdvice" transaction-manager="transactionManager">
2
<tx:attributes>
3
<tx:method name="purchase" propagation="REQUIRES_NEW"
4
isolation="READ_COMMITTED"/>
5
</tx:attributes>
6
</tx:advice>
Copied!

7.设置回滚事务属性

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

8.超时和只读属性

  • 由于事务可以在行和表上获得锁,因此长事务会占用资源,并对整体性能产生影响。
  • 如果一个事务只读取数据但不做修改,数据库引擎可以对这个事务进行优化。
  • 超时事务属性:事务在强制回滚之前可以保持多久。这样可以防止长期运行的事务占用资源。
  • 只读事务属性:表示这个事务只读取数据库但不更新数据库,这样可以帮助数据库引擎优化事务。
  • 超时和只读属性可以在@Transactional注解中定义,超时属性以秒为单位。
    1
    @Transactional(propagation=Propagation.REQUIRED_NEW,
    2
    isolation=Isolation.READ_COMMITTED,
    3
    rollbackFor=IOException.class,SQLException.class,
    4
    noRollbackFor=ArithmeticException.class,
    5
    readOnly=true,tomeout=30)
    6
    public void purchase(String isbn,String username){}
    Copied!
  • 在Spring 2.x事务通知中,超时和只读属性可以在元素中进行指定
    1
    <tx:advice id="bookShopTxAdvice" transaction-manager="transactionManager">
    2
    <tx:attributes>
    3
    <tx:method name="purchase" propagation="REQUIRES_NEW"
    4
    isolation="READ_COMMITTED"
    5
    rollback-for="java.io.IOException,java.sql.SQLException"
    6
    no-rollback-for="java.lang,ArithmeticException"
    7
    read-only=true
    8
    timeout=30/>
    9
    </tx:attributes>
    10
    </tx:advice>
    Copied!
Last modified 2yr ago