Spring应用——事务管理

事务基础:请参看:http://www.cnblogs.com/solverpeng/p/5720306.html

一、Spring 事务管理

1.前提:事务管理器

在使用 Spring 声明式事务管理策略之前,必须配置事务管理器。

Spring 的核心事务管理器的顶级接口是 PlatformTransactionManager。

DataSourceTransactionManager:在应用程序中只需要处理一个数据源,而且通过 JDBC 进行存取。

HibernateTransactionManager:用 Hibernate 框架存取数据。

事务管理器以普通 Bean 的形式生命在 Spring IOC 容器中。如:

<context:property-placeholder location="db.properties"/>

<bean class="com.mchange.v2.c3p0.ComboPooledDataSource" id="dataSource">
  <property name="user" value="${jdbc.user}"/>
  <property name="password" value="${jdbc.password}"/>
  <property name="jdbcUrl" value="${jdbc.jdbcUrl}"/>
  <property name="driverClass" value="${jdbc.DriverClass}"/>
</bean>

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

2.XML声明式事务管理:Spring 通过 SpringAOP 框架支持声明式事务。

(1)事务管理是一个横切关注点。

(2)具体操作:

<1>在 <beans> 根元素中添加  tx Schema 的约束。可以通过 tx  Schema 中定义的 <tx:advice> 元素声明事务增强。

<tx:advice transaction-manager="transactionManager" id="txAdvice"/>

<2>将增强配置到相应的 Spring AOP 切面。

<aop:config>
  <aop:pointcut id="bookPointcut" expression="execution(* *.BookServiceImpl.*(..))"/>
  <aop:advisor advice-ref="txAdvice" pointcut-ref="bookPointcut"/>
</aop:config>

需要注意的是:只有公有的方法才可以添加增强。

3.注解声明式的管理事务:Spring 通过 @Transacational 注解声明式地管理事务

(1)Spring 允许使用 @Transacational 注解来标注事务方法。只能标注公有方法。

(2)可以在方法或类级别添加 @Transactional 注解。当添加到类上时,这个类的所有公有方法都会被定义成支持事务处理的。

(3)Spring Config 文件中的配置:

<1>只需要添加 <tx:annotation-driven/> 节点,并为其指定事务管理器就好了。

<2>指定事务管理器:若事务管理器名称是 transacationManager,就可以在 <tx:annotation-driven/> 节点中省略 transaction-manager 属性。

该元素会自动检测该名称的事务管理器。

<context:property-placeholder location="db.properties"/>

<bean class="com.mchange.v2.c3p0.ComboPooledDataSource" id="dataSource">
  <property name="user" value="${jdbc.user}"/>
  <property name="password" value="${jdbc.password}"/>
  <property name="jdbcUrl" value="${jdbc.jdbcUrl}"/>
  <property name="driverClass" value="${jdbc.DriverClass}"/>
</bean>

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

<tx:annotation-driven/>

二、Spring 事务传播行为

1.JDBC 提供了事务隔离级别这种解决方案,Spring 对其进行了补充和扩展,就是 事务的传播行为。

2.Spring 提供了七种事务传播行为:

PROPAGATION_REQUIREDPROPAGATION_REQUIRED_NEW、PROPAGATION_NESTED、PROPAGATION_SUPPORTS、PROPAGATION_NOT_SUPPORTED、PROPAGATION_NEVER、PROPAGATION_MANDATORY

我自己理解的 PROPAGATION_REQUIRED 和 PROPAGATION_REQUIRED_NEW :

方法 A 调用方法 B:

(1)如果 A 是一个事务方法,B 也是一个事务方法,那么 B 应该使用自己的事务还是 A 的事务?

如果使用 A 的事务 —— PROPAGATION_REQUIRED

如果使用 B 自己的事务 —— PROPAGATION_REQUIRED_NEW 

举个例子来说明:

一个事务:A、B两位同学去饭馆吃饭

中午放学后,A 同学去饭馆吃饭,恰巧刚刚认识的 B 同学也在饭馆吃饭,A 同学是该自己吃呢?还是和 B 同学一起吃呢?

如果是和 B 一起 —— PROPAGATION_REQUIRED

如果是 A 自己吃 —— PROPAGATION_REQUIRED_NEW

(2)如果 A 是一个事务方法,B 不是一个事务方法,那么 B 会使用 A 的事务。

提示:PROPAGATION 意思是 传播

*上面是我自己的理解,没有将所有的情况覆盖,可能自己理解的也有偏差,看到此处的时候请小心求证。

(3)Spring 默认的事务传播行为为 PROPAGATION_REQUIRED

三、Spring 事务其他功能

除了强大的事务传播行为外,Spring 还提供了一些小的附加功能

1.事务超时——为了解决事务时间太长,消耗资源太多的问题,给事务设置一个最大时长,如果超时,则回滚事务。以 秒为单位。

2.只读事务——表示这个事务只读取数据而不更新数据

3.设置事务的回滚属性

默认情况下只检查运行时异常,会导致事务回滚。可以通过 rollbackFor 属性来定义,如果不止一种,可以通过 "," 进行分割。

不论是事务的传播行为还是事务的超时和只读属性,Spring 在 @Transactional 注解中提供了对应的属性。

也可以通过 XML 的方式去配置,如

<tx:advice transaction-manager="transactionManager" id="bookAdvice">
  <tx:attributes>
    <tx:method name="shop" propagation="REQUIRED"/>
  </tx:attributes>
</tx:advice>

四、Spring 事务处理时,自我调用事务不生效。

问题描述:

@Service
public class PersonService {
    @Autowired
    private PersonDao personDao;

    @Transactional
    public void outter() {
        for (int i = 1; i <= 10; i++) {
            this.inner(i);
        }
    }

    @Transactional(propagation= Propagation.REQUIRES_NEW)
    public void inner(int index) {
        personDao.insert(index);
        if(index==3){
            int ii = 0;
            ii = 1/0;
            System.out.println(ii);
        }
    }

}

Spring Config 文件

<context:component-scan base-package="com.nucsoft.spring"/>
<context:property-placeholder location="classpath:db.properties"/>

<bean class="com.mchange.v2.c3p0.ComboPooledDataSource" id="dataSource">
    <property name="driverClass" value="${jdbc.driverClass}"/>
    <property name="jdbcUrl" value="${jdbc.jdbcUrl}"/>
    <property name="user" value="${jdbc.user}"/>
    <property name="password" value="${jdbc.password}"/>
</bean>

<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    <property name="dataSource" ref="dataSource"/>
</bean>

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

<tx:annotation-driven/>

测试:

@Test
public void testSupport() {
    PersonService bean = context.getBean(PersonService.class);
    bean.outter();
}

结果:不论是将inner() 方法的 propagation 改为 Propagation.REQUIRED 还是 Propagation.REQUIRES_NEW,数据库中都没有插入数据。

分析:

因为 Spring 的事务管理是基于动态代理的,如果 target 中的方法不是被代理对象调用的,那么就不会织入切面代码,即事务不会生效。

对于上面的例子,在 outter() 方法中调用 inner() 方法的 this 是 PersonService 类对象。而不是 PersonService 的代理对象。

解决方案:

使用代理对象调用 inner() 方法。

具体做法:

(1)在 Spring Config 文件中添加如下配置:

<aop:aspectj-autoproxy expose-proxy="true"/>

说明:是否允许暴露代理对象。expose-proxy就是向ThreadLocal中存代理对象。

(2)手动的暴露代理对象。

@Transactional
public void outter() {
    for (int i = 1; i <= 10; i++) {
        ((PersonService) AopContext.currentProxy()).inner(i);
    }
}

@Transactional(propagation= Propagation.REQUIRES_NEW)
public void inner(int index) {
    personDao.insert(index);
    if(index==3){
        int ii = 0;
        ii = 1/0;
        System.out.println(ii);
    }
}

说明:AopContext.currentProxy就是从ThreadLocal中取出代理对象。

参考的文章:

http://blog.csdn.net/derrantcm/article/details/46284811

http://jinnianshilongnian.iteye.com/blog/1487235

 

posted @ 2016-07-30 15:07  solverpeng  阅读(1222)  评论(0编辑  收藏  举报