一、事务的作用
将若干的数据库操作作为一个整体控制,一起成功或一起失败。
原子性:指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。
一致性:指事务前后数据的完整性必须保持一致。
隔离性:指多个用户并发访问数据库时,一个用户的事务不能被其他用户的事务所干扰,多个并发事务之间数据要相互隔离。
持久性:指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,即时数据库发生故障也不应该对其有任何影响。
二、Spring事务管理高层抽象主要包括3个接口
--Platform TransactionManager 事务管理器(提交、回滚事务)
Spring为不同的持久化框架提供了不同的Platform TransactionManager接口实现。如:
使用Spring JDBC或iBatis进行持久化数据时使用DataSourceTransactionManager
使用Hibernate3.0版本进行持久化数据时使用HibernateTransactionManager
--TransactionDefinition 事务定义信息(隔离、传播、超时、只读)
脏读:一个事务读取了另一个事务改写但还未提交的数据,如果这些数据被回滚,则读到的数据是无效的。
不可重复读:在同一事务中,多次读取同一数据返回的结果有所不同。
幻读:一个事务读取了几行记录后,另一个事务插入一些记录,幻读就发生了。再后来的查询中,第一个事务就会发现有些原来没有的记录。
事务隔离级别:(五种)
- DEFAULT--使用后端数据库默认的隔离级别(Spring中的选择项)
- READ_UNCOMMITED--允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读
- READ_COMMITTED--允许在并发事务已经提交后读取。可防止脏读,但幻读和不可重复读仍可发生
- REPEATABLE_READ--对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生
- SERIALIZABLE--完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的
其中,MySQL默认采用REPEATABLE_READ隔离级别;Oracle默认采用READ_COMMITTED隔离级别
事务传播行为:(七种)
- REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
- SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。
- MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。
- REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。
- NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
- NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。
- NESTED--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与REQUIRED类似的操作。拥有多个可以回滚的保存点,内部回滚不会对外部事务产生影响。只对DataSourceTransactionManager有效
--TransactionStatus 事务具体运行状态
三、Spring提供了以下方法控制事务
a.编程式事务管理(基于Java编程控制,很少使用)--见demo1包
利用TransactionTemplate将多个DAO操作封装起来
*b.声明式事务管理(基于Spring的AOP配置控制)
-基于TransactionProxyFactoryBean的方式.(很少使用)--见demo2包
需要为每个进行事务管理的类,配置一个TransactionProxyFactoryBean进行增强.
-基于XML配置(经常使用)--见demo3包
一旦配置好之后,类上不需要添加任何东西。
如果Action作为目标对象切入事务,需要在<aop:config>元素里添加proxy-target-class="true"属性。原因是通知Spring框架采用CGLIB技术生成具有事务管理功能的Action类。
-基于注解(配置简单,经常使用)--见demo4包
在applicationContext.xml中开启事务注解配置。(applicationContext.xml中只需定义Bean并追加以下元素)
<bean id="txManager" class="...">
<property name="sessionFactory">
</property>
<tx:annotation-driven transaction-manager="txManager"/>
在目标组件类中使用@Transactional,该标记可定义在类前或方法前。
四、示例(银行转账)
--编程式
-
/** -
* @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); -
}
-
/** -
* @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); -
} -
}
-
/** -
* @Description:转账案例的业务接口 -
* -
*/ -
public interface AccountService { -
/** -
* @param out :转出账号 -
* @param in :转入账号 -
* @param money :转账金额 -
*/ -
public void transfer(String out,String in,Double money); -
}
-
/** -
* @Description:转账案例的业务层实现类 -
*/ -
public class AccountServiceImpl implements AccountService { -
// 注入转账的DAO -
private AccountDao accountDao; -
// 注入事务管理的模板 -
private TransactionTemplate transactionTemplate; -
/** -
* @param out -
* :转出账号 -
* @param in -
* :转入账号 -
* @param money -
* :转账金额 -
*/ -
@Override -
public void transfer(final String out, final String in, final Double money) { -
// 未经事务控制的业务处理操作,如果过程中出异常,则导致前面的操作能完成,后面的不能,即转账成功但未收到转账款 -
// accountDao.outMoney(out, money); -
// int i = 1/0; -
// accountDao.inMoney(in, money); -
transactionTemplate.execute(new TransactionCallbackWithoutResult() { -
@Override -
protected void doInTransactionWithoutResult( -
TransactionStatus transactionStatus) { -
accountDao.outMoney(out, money); -
// int i = 1 / 0;//事务控制,即出现异常,该段内代码都执行失效 -
accountDao.inMoney(in, money); -
} -
}); -
} -
public void setAccountDao(AccountDao accountDao) { -
this.accountDao = accountDao; -
} -
public void setTransactionTemplate(TransactionTemplate transactionTemplate) { -
this.transactionTemplate = transactionTemplate; -
} -
}
applicationContext1.xml
-
<!-- 引入外部的属性文件 --> -
<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.zs.spring.demo1.AccountServiceImpl"> -
<property name="accountDao" ref="accountDao" /> -
<!-- 注入事务管理的模板 --> -
<property name="transactionTemplate" ref="transactionTemplate" /> -
</bean> -
<!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> -
<bean id="accountDao" class="com.zs.spring.demo1.AccountDaoImpl"> -
<property name="dataSource" ref="dataSource" /> -
</bean> -
<!-- 配置DAO类(未简化) --> -
<!-- <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> -
<property name="dataSource" ref="dataSource" /> -
</bean> -
<bean id="accountDao" class="com.zs.spring.demo1.AccountDaoImpl"> -
<property name="jdbcTemplate" ref="jdbcTemplate" /> -
</bean> --> -
<!-- ==================================1.编程式的事务管理=============================================== --> -
<!-- 配置事务管理器 --> -
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> -
<property name="dataSource" ref="dataSource" /> -
</bean> -
<!-- 配置事务管理的模板:Spring为了简化事务管理的代码而提供的类 --> -
<bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate"> -
<property name="transactionManager" ref="transactionManager"/> -
</bean>
测试:
-
@RunWith(SpringJUnit4ClassRunner.class) -
@ContextConfiguration("classpath:applicationContext1.xml") -
public class TransactionTest { -
@Resource(name = "accountService") -
private AccountService accountService; -
@Test -
public void demo1() { -
accountService.transfer("aaa", "bbb", 200d); -
} -
}
--基于TransactionProxyFactoryBean的方式
-
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; -
} -
}
applicationContext2.xml
-
<!-- 引入外部的属性文件 --> -
<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.zs.spring.demo2.AccountServiceImpl"> -
<property name="accountDao" ref="accountDao" /> -
</bean> -
<!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> -
<bean id="accountDao" class="com.zs.spring.demo2.AccountDaoImpl"> -
<property name="dataSource" ref="dataSource" /> -
</bean> -
<!-- ==================================2.使用XML配置声明式的事务管理(原始方式)=============================================== --> -
<!-- 配置事务管理器 --> -
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> -
<property name="dataSource" ref="dataSource" /> -
</bean> -
<!-- 配置业务层的代理 --> -
<bean id="accountServiceProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"> -
<!-- 配置目标对象 --> -
<property name="target" ref="accountService" /> -
<!-- 注入事务管理器 --> -
<property name="transactionManager" ref="transactionManager"></property> -
<!-- 注入事务的属性 --> -
<property name="transactionAttributes"> -
<props> -
<!-- -
prop的格式: -
* PROPAGATION :事务的传播行为 -
* ISOTATION :事务的隔离级别 -
* readOnly :只读 -
* -EXCEPTION :发生哪些异常回滚事务 -
* +EXCEPTION :发生哪些异常不回滚事务 -
--> -
<prop key="transfer">PROPAGATION_REQUIRED</prop> -
<!-- <prop key="transfer">PROPAGATION_REQUIRED,readOnly</prop> --> -
<!-- <prop key="transfer">PROPAGATION_REQUIRED,+java.lang.ArithmeticException</prop> --> -
</props> -
</property> -
</bean>
测试:
-
@RunWith(SpringJUnit4ClassRunner.class) -
@ContextConfiguration("classpath:applicationContext2.xml") -
public class TransactionTest { -
/** -
* 一定要注入代理类:因为代理类进行增强的操作 -
*/ -
// @Resource(name="accountService") -
@Resource(name = "accountServiceProxy") -
private AccountService accountService; -
@Test -
public void demo1() { -
accountService.transfer("aaa", "bbb", 200d); -
} -
}
--基于XML配置
-
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; -
} -
}
applicationContext3.xml
-
<!-- 引入外部的属性文件 --> -
<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.zs.spring.demo3.AccountServiceImpl"> -
<property name="accountDao" ref="accountDao" /> -
</bean> -
<!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> -
<bean id="accountDao" class="com.zs.spring.demo3.AccountDaoImpl"> -
<property name="dataSource" ref="dataSource" /> -
</bean> -
<!-- ==================================3.使用XML配置声明式的事务管理,基于tx/aop=============================================== --> -
<!-- 配置事务管理器 --> -
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> -
<property name="dataSource" ref="dataSource" /> -
</bean> -
<!-- 配置事务的通知 --> -
<tx:advice id="txAdvice" transaction-manager="transactionManager"> -
<tx:attributes> -
<!-- -
propagation :事务传播行为 -
isolation :事务的隔离级别 -
read-only :只读 -
rollback-for:发生哪些异常回滚 -
no-rollback-for :发生哪些异常不回滚 -
timeout :过期信息 -
--> -
<tx:method name="transfer" propagation="REQUIRED"/> -
</tx:attributes> -
</tx:advice> -
<!-- 配置切面 --> -
<aop:config> -
<!-- 配置切入点 --> -
<aop:pointcut expression="execution(* com.zs.spring.demo3.AccountService+.*(..))" id="pointcut1"/> -
<!-- 配置切面 --> -
<aop:advisor advice-ref="txAdvice" pointcut-ref="pointcut1"/> -
</aop:config>
测试:
-
/** -
* @Description:Spring的声明式事务管理的方式二:基于AspectJ的XML方式的配置 -
*/ -
@RunWith(SpringJUnit4ClassRunner.class) -
@ContextConfiguration("classpath:applicationContext3.xml") -
public class TransactionTest { -
/** -
* 一定要注入代理类:因为代理类进行增强的操作 -
*/ -
@Resource(name = "accountService") -
private AccountService accountService; -
@Test -
public void demo1() { -
accountService.transfer("aaa", "bbb", 200d); -
} -
}
--基于注解
-
/** -
* @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; -
} -
}
applicationContext4.xml
-
<!-- 引入外部的属性文件 --> -
<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.zs.spring.demo4.AccountServiceImpl"> -
<property name="accountDao" ref="accountDao" /> -
</bean> -
<!-- 配置DAO类(简化,会自动配置JdbcTemplate) --> -
<bean id="accountDao" class="com.zs.spring.demo4.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"/>
测试:
-
@RunWith(SpringJUnit4ClassRunner.class) -
@ContextConfiguration("classpath:applicationContext4.xml") -
public class TransactionTest { -
/** -
* 一定要注入代理类:因为代理类进行增强的操作 -
*/ -
@Resource(name = "accountService") -
private AccountService accountService; -
@Test -
public void demo1() { -
accountService.transfer("aaa", "bbb", 200d); -
} -
}
转载至:http://blog.youkuaiyun.com/daijin888888/article/details/51822257
本文详细介绍了Spring事务管理的原理及实现方式,包括编程式、声明式事务管理等,并通过转账案例展示了不同配置方法的应用。
2630

被折叠的 条评论
为什么被折叠?



