Spring事务管理

什么是事务

可以理解为每次与数据库交互的过程就是一个事务。事务是将若干的数据库操作作为一个整体控制,一起成功或一起失败。

事务的四大特性

  1. 原子性:指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。
  2. 一致性:指事务前后数据的完整性必须保持一致。
  3. 隔离性:指多个用户并发访问数据库时,一个用户的事务不能被其他用户的事务所干扰,多个并发事务之间数据要相互隔离。
  4. 持久性:指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,即时数据库发生故障也不应该对其有任何影响。

事务的隔离级别

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

事务的传播行为

  1. REQUIRED–支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
  2. SUPPORTS–支持当前事务,如果当前没有事务,就以非事务方式执行。
  3. MANDATORY–支持当前事务,如果当前没有事务,就抛出异常。
  4. REQUIRES_NEW–新建事务,如果当前存在事务,把当前事务挂起。
  5. NOT_SUPPORTED–以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
  6. NEVER–以非事务方式执行,如果当前存在事务,则抛出异常。
  7. NESTED–如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与REQUIRED类似的操作。拥有多个可以回滚的保存点,内部回滚不会对外部事务产生影响。只对DataSourceTransactionManager有效

基于注解实现事务

DAO

/** 
 * @Description:转账案例的DAO层接口 
 *  
 */  
public interface AccountDao {  
    /** 
     * @param out 
     *            :转出账号 
     * @param money 
     *            :转账金额 
     */  
    public void outMoney(String out, Double money);  

    /** 
     *  
     * @param in 
     *            :转入账号 
     * @param money 
     *            :转账金额 
     */  
    public void inMoney(String in, Double money);  
}  

DAOImpl

/** 
 * @Description:转账案例的DAO层实现类 
 */  
public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {  
    /** 
     * @param out 
     *            :转出账号 
     * @param money 
     *            :转账金额 
     */  
    @Override  
    public void outMoney(String out, Double money) {  
        String sql = "update account set money = money-? where name = ?";  
        this.getJdbcTemplate().update(sql, money, out);  
    }  
    /** 
     * @param in 
     *            :转入账号 
     * @param money 
     *            :转账金额 
     */  
    @Override  
    public void inMoney(String in, Double money) {  
        String sql = "update account set money = money+? where name = ?";  
        this.getJdbcTemplate().update(sql, money, in);  
    }  
} 

Service

/** 
 * @Description:转账案例的业务接口 
 * 
 */  
public interface AccountService {  
    /** 
     * @param out   :转出账号 
     * @param in    :转入账号 
     * @param money :转账金额 
     */  
    public void transfer(String out,String in,Double money);  
} 
<!-- 引入外部的属性文件 -->  
    <context:property-placeholder location="classpath:jdbc.properties"/>  

    <!-- 配置c3p0连接池 -->  
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">  
        <property name="driverClass" value="${jdbc.driverClass}" />  
        <property name="jdbcUrl" value="${jdbc.url}" />  
        <property name="user" value="${jdbc.username}" />  
        <property name="password" value="${jdbc.password}" />  
    </bean>  

    <!-- 配置业务层类 -->  
    <bean id="accountService" class="com.spring.demo.AccountServiceImpl">  
        <property name="accountDao" ref="accountDao" />  
    </bean>  

    <!-- 配置DAO类(简化,会自动配置JdbcTemplate) -->  
    <bean id="accountDao" class="com.spring.demo.AccountDaoImpl">  
        <property name="dataSource" ref="dataSource" />  
    </bean>  

    <!-- ==================================4.使用注解配置声明式事务============================================ -->  

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

    <!-- 开启注解事务 -->  
    <tx:annotation-driven transaction-manager="transactionManager"/>  
/** 
 * @Transactional中的的属性 propagation :事务的传播行为 isolation :事务的隔离级别 readOnly :只读 
 *                     rollbackFor :发生哪些异常回滚 noRollbackFor :发生哪些异常不回滚 
 *                     rollbackForClassName 根据异常类名回滚 
 */  
@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false)  
public class AccountServiceImpl implements AccountService {  
    // 注入转账的DAO  
    private AccountDao accountDao;  

    /** 
     * @param out 
     *            :转出账号 
     * @param in 
     *            :转入账号 
     * @param money 
     *            :转账金额 
     */  
    @Override  
    public void transfer(String out, String in, Double money) {  
        accountDao.outMoney(out, money);  
        // int i = 1/0;  
        accountDao.inMoney(in, money);  
    }  

    public void setAccountDao(AccountDao accountDao) {  
        this.accountDao = accountDao;  
    }  
}  

猜你喜欢

转载自blog.csdn.net/wangchengming1/article/details/80005941