在上一篇的最后,我们引出了问题的产生,接下来看看Spring中是如何解决此类问题的。

一、Spring中事务属性的介绍

       Spring中事务的属性有以下几种:

(1)事务传播行为(propagation)

当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。Spring定义了七种传播行为,如下表:

传播行为


描述

PROPAGATION_REQUIRED

required

表示当前方法必须运行在事务中。如果当前事务存在,方法将会在该事务中运行。否则就开启一个新的事务

PROPAGATION_SUPPORTS

supports

表示当前方法不需要事务上下文,但是如果存在当前事务的话,那么该方法会在这个事务中运行。

PROPAGATION_MANDATORY

mandatory

表示该方法必须在事务中运行,如果当前事务不存在,则抛出一个异常。

PROPAGATION_REQUIRED_NEW

required_new

表示当前方法必须运行在它自己的事务中。一跟新的事务会被启动。如果存在当前事务,在该方法执行期间,原事务被挂起。如果使用JATransactionManager的话,则需要访问TransactionManager

PROPAGATION_NOT_SUPPORTED

not_supported

表示该方法不应该运行在事务中。如果存在当前事务,在该方法执行期间,原事务被挂起。如果使用JATransactionManager的话,则需要访问TransactionManager

PROPAGATION_NEVER

never

表示当前方法不应该运行在事务上下文中。如果当前正有一个事务在运行,则抛出异常

PROPAGATION_NESTED

nested

表示如果当前已经存在一个事务,那么该方法将会在嵌套事务中运行。嵌套的事务可以独立于当前事务进行单独地提交或回滚。如果当前事务不存在,那么其行为与PROPAGATION_REQUIRED一样

(2)事务隔离级别(isolation)

Spring中的五种事务隔离级别:
DEFAULT:使用后端数据库默认的隔离级别(spring中的的选择项)
READ_UNCOMMITED:允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读。
READ_COMMITTED:允许在并发事务已经提交后读取。可防止脏读,但是读和不可重复读仍可发生。
REPEATABLE_READ:对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生。
SERIALIZABLE:完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的。

(3)超时时间(timeout)

事务需要在一定时间内提交,如果不提交就进行回滚。默认值是-1,设置时间以秒为单位。

(4)是否只读(readOnly)

默认值为false,表示可以查询、修改、删除、添加等操作。读指的是查询操作,写指的是添加修改删除操作。

(5)回滚(rollbackFor)

设置出现哪些异常事务回滚。

(6)不回滚(noRollbackFor)

设置出现哪些异常事务不进行回滚。

二、编程式事务管理

       编程式事务指的是通过编码方式实现事务,即类似于 JDBC 编程实现事务管理。管理使用 TransactionTemplate 或者直接使用底层的 PlatformTransactionManager。对于编程式事务管理,spring 推荐使用 TransactionTemplate。

(1)bean.xml中配置transactionTemplate和事务管理器

<!--  配置transactionTemplate -->
    <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
        <property name="transactionManager" ref="transactionManager"></property>
    </bean>

    <!--  创建事务管理器  -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>

(2) 在tranferAccount()方法中通过transactionTemplate.execute执行逻辑,通过TransactionCallback接口中的方法后将返回值传递到TransactionTemplate的execute()中。通过调用TransactionStatus 的setRollbackOnly()方法来实现事物回滚。

@Service
public class UserService {
    @Autowired
    private IUserDao userDao;
    @Autowired
    private TransactionTemplate transactionTemplate;
    public void tranferAccount(){
        transactionTemplate.execute(new TransactionCallback<String>() {

            @Override
            public String doInTransaction(TransactionStatus transactionStatus) {
                try{
                    userDao.reduceMoney();
                    int i = Integer.parseInt("123a");
                    userDao.addMoney();
                }catch (Exception e){
                    TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); //手动开启事务回滚
                    transactionStatus.setRollbackOnly();
                    return null;
                }
                return null;
            }
        });

    }
}

三、声明式事务管理

       声明式事务管理建立在 AOP 之上的。其本质是对方法前后进行拦截,然后目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。声明式事务最大的优点就是不需要通过编程的方式管理事务,这样就不需要在业务逻辑代码中掺杂事务管理的代码,只需在配置文件中做相关的事务规则声明 (或通过基于 @Transactional 注解的方式),便可以将事务规则应用到业务逻辑中

1、基于XML的声明式事务管理

<!--  创建事务管理器  -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!-- 编写通知:对事务管理器进行增强(通知),编写切入点和具体执行事务的细节   -->
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <!--    给切入点方法添加事务详情        -->
            <tx:method name="tranferAccount" propagation="REQUIRED" isolation="READ_COMMITTED"/>
            <tx:method name="*" read-only="false"/>
        </tx:attributes>
    </tx:advice>
    <!--  Aop编写:让Spring自动对目标生成代理,需要使用 AspectJ表达式 -->
    <aop:config>
        <!--    切入点    -->
        <aop:pointcut id="pt" expression="execution(* com.yht.example8.service.UserService.*(..))"/>
        <!--   切面:将切入点和通知整合     -->
        <aop:advisor advice-ref="txAdvice" pointcut-ref="pt"></aop:advisor>
    </aop:config>

2、基于注解的声明式事务管理

(1)在bean.xml中开启事务注解

<!--  开启事务注解  -->
    <tx:annotation-driven></tx:annotation-driven>

(2)修改UserService类,添加注解@Transactional

@Service
@Transactional
public class UserService {
    @Autowired
    private IUserDao userDao;
    @Autowired
    private TransactionTemplate transactionTemplate;
    public void tranferAccount(){
        userDao.reduceMoney();
        int i = Integer.parseInt("123a");
        userDao.addMoney();
    }
}

说明:完全注解方式实现声明式事务管理,在config下提供一个MyAopConfig类,如下:

@Configuration    //配置类
@ComponentScan("com.yht.example8")      //组件扫描
@EnableAspectJAutoProxy     //开启事务
public class MyAopConfig {
    // @Bean:用于把当前方法的返回值作为bean对象存入spring的ioc容器中

    /**
     * 获取数据库连接池
     * @return
     */
    @Bean
    public DruidDataSource getDruidDataSource(){
        DruidDataSource dataSource = new DruidDataSource();
        dataSource.setDriverClassName("com.mysql.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://localhost:3306/test");
        dataSource.setUsername("root");
        dataSource.setPassword("root");
        return dataSource;
    }

    /**
     * 获取JdbcTemplate
     * @param dataSource 由IOC容器提供
     * @return
     */
    @Bean
    public JdbcTemplate getJdbcTemplate(DataSource dataSource){
        JdbcTemplate jdbcTemplate = new JdbcTemplate();
        //注入dataSource
        jdbcTemplate.setDataSource(dataSource);
        return jdbcTemplate;
    }
}