Spring事务管理

Spring支持两种方式的事务管理:

  • 编程式事务管理: 通过Transaction Template手动管理事务,实际应用中很少使用,
  • 使用XML配置声明式事务: 推荐使用(代码侵入性最小),实际是通过AOP实现

实现声明式事务的四种方式:

  1. 基于 TransactionInterceptor 的声明式事务: Spring 声明式事务的基础,通常也不建议使用这种方式,但是与前面一样,了解这种方式对理解 Spring 声明式事务有很大作用。
  2. 基于 TransactionProxyFactoryBean 的声明式事务: 第一种方式的改进版本,简化的配置文件的书写,这是 Spring 早期推荐的声明式事务管理方式,但是在 Spring 2.0 中已经不推荐了。
  3. 基于< tx> 和< aop>命名空间的声明式事务管理: 目前推荐的方式,其最大特点是与 Spring AOP 结合紧密,可以充分利用切点表达式的强大支持,使得管理事务更加灵活。
  4. 基于 @Transactional 的全注解方式: 将声明式事务管理简化到了极致。开发人员只需在配置文件中加上一行启用相关后处理 Bean 的配置,然后在需要实施事务管理的方法或者类上使用 @Transactional 指定事务规则即可实现事务管理,而且功能也不必其他方式逊色。

我们今天要将的是使用编程式以及基于AspectJ的声明式和基于注解的事务方式,实现烂大街的转账业务。

再来说一下这个案例的思想吧,我们在两次转账之间添加一个错误语句(对应银行断电等意外情况),如果这个时候两次转账不能成功,则说明事务配置正确,否则,事务配置不正确。

你需要完成的任务:

  • 使用编程式事务管理完成转账业务
  • 使用基于AspectJ的声明式事务管理完成转账业务
  • 使用基于 @Transactional 的全注解方式事务管理完成转账业务

备注:

下面的代码是在很久之前,我刚学Sping还没有接触Maven的时候写的,所以我使用的原始添加jar的方式,使用Maven的小伙伴可以自行添加Maven依赖,没有使用Maven的小伙伴直接使用我下面提供的jar包即可。

jar包地址:链接:https://pan.baidu.com/s/1tqy-mVKxSutsIIvYgtC3Rw 密码:nid0

项目结构:
Spring 编程式和声明式事务实例 - 图1

开发工具:

Myeclipse2017

SQL:

  1. create table `account` (
  2. `username` varchar (99),
  3. `salary` int (11)
  4. );
  5. insert into `account` (`username`, `salary`) values('小王','3000');
  6. insert into `account` (`username`, `salary`) values('小马','3000');

(1)编程式事务管理

注意: 通过添加/删除accountMoney() 方法中int i = 10 / 0这个语句便可验证事务管理是否配置正确。

OrdersDao.java(Dao层)

  1. package cn.itcast.dao;
  2. import org.springframework.jdbc.core.JdbcTemplate;
  3. public class OrdersDao {
  4. // 注入jdbcTemplate模板对象
  5. private JdbcTemplate jdbcTemplate;
  6. public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
  7. this.jdbcTemplate = jdbcTemplate;
  8. }
  9. // 对数据操作的方法不包含业务操作
  10. /**
  11. * 小王少钱的方法
  12. */
  13. public void reduceMoney() {
  14. String sql = "update account set salary=salary-? where username=?";
  15. jdbcTemplate.update(sql, 1000, "小王");
  16. }
  17. /**
  18. * 小马多钱的方法
  19. */
  20. public void addMoney() {
  21. String sql = "update account set salary=salary+? where username=?";
  22. jdbcTemplate.update(sql, 1000, "小马");
  23. }
  24. }

OrdersService.java(业务逻辑层)

  1. package cn.itcast.service;
  2. import org.springframework.transaction.TransactionStatus;
  3. import org.springframework.transaction.support.TransactionCallback;
  4. import org.springframework.transaction.support.TransactionTemplate;
  5. import cn.itcast.dao.OrdersDao;
  6. public class OrdersService {
  7. // 注入Dao层对象
  8. private OrdersDao ordersDao;
  9. public void setOrdersDao(OrdersDao ordersDao) {
  10. this.ordersDao = ordersDao;
  11. }
  12. // 注入TransactionTemplate对象
  13. private TransactionTemplate transactionTemplate;
  14. public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
  15. this.transactionTemplate = transactionTemplate;
  16. }
  17. // 调用dao的方法
  18. // 业务逻辑,写转账业务
  19. public void accountMoney() {
  20. transactionTemplate.execute(new TransactionCallback<Object>() {
  21. @Override
  22. public Object doInTransaction(TransactionStatus status) {
  23. Object result = null;
  24. try {
  25. // 小马多1000
  26. ordersDao.addMoney();
  27. // 加入出现异常如下面int
  28. // i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱
  29. // 解决办法是出现异常后进行事务回滚
  30. int i = 10 / 0;// 事务管理配置后异常已经解决
  31. // 小王 少1000
  32. ordersDao.reduceMoney();
  33. } catch (Exception e) {
  34. status.setRollbackOnly();
  35. result = false;
  36. System.out.println("Transfer Error!");
  37. }
  38. return result;
  39. }
  40. });
  41. }
  42. }

TestService.java(测试方法)

  1. package cn.itcast.service;
  2. import org.junit.Test;
  3. import org.springframework.context.ApplicationContext;
  4. import org.springframework.context.support.ClassPathXmlApplicationContext;
  5. public class TestService {
  6. @Test
  7. public void testAdd() {
  8. ApplicationContext context = new ClassPathXmlApplicationContext(
  9. "beans.xml");
  10. OrdersService userService = (OrdersService) context
  11. .getBean("ordersService");
  12. userService.accountMoney();
  13. }
  14. }

配置文件:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
  4. xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
  5. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
  6. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
  7. http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
  8. http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
  9. <!-- 配置c3po连接池 -->
  10. <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
  11. <!-- 注入属性值 -->
  12. <property name="driverClass" value="com.mysql.jdbc.Driver"></property>
  13. <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
  14. <property name="user" value="root"></property>
  15. <property name="password" value="153963"></property>
  16. </bean>
  17. <!-- 编程式事务管理 -->
  18. <!-- 配置事务管理器 -->
  19. <bean id="dataSourceTransactionManager"
  20. class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  21. <!-- 注入dataSource -->
  22. <property name="dataSource" ref="dataSource"></property>
  23. </bean>
  24. <!-- 配置事务管理器模板 -->
  25. <bean id="transactionTemplate"
  26. class="org.springframework.transaction.support.TransactionTemplate">
  27. <!-- 注入真正进行事务管理的事务管理器,name必须为 transactionManager否则无法注入 -->
  28. <property name="transactionManager" ref="dataSourceTransactionManager"></property>
  29. </bean>
  30. <!-- 对象生成及属性注入 -->
  31. <bean id="ordersService" class="cn.itcast.service.OrdersService">
  32. <property name="ordersDao" ref="ordersDao"></property>
  33. <!-- 注入事务管理的模板 -->
  34. <property name="transactionTemplate" ref="transactionTemplate"></property>
  35. </bean>
  36. <bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
  37. <property name="jdbcTemplate" ref="jdbcTemplate"></property>
  38. </bean>
  39. <!-- JDBC模板对象 -->
  40. <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
  41. <property name="dataSource" ref="dataSource"></property>
  42. </bean>
  43. </beans>

(2)基于AspectJ的声明式事务管理

OrdersService.java(业务逻辑层)

  1. package cn.itcast.service;
  2. import cn.itcast.dao.OrdersDao;
  3. public class OrdersService {
  4. private OrdersDao ordersDao;
  5. public void setOrdersDao(OrdersDao ordersDao) {
  6. this.ordersDao = ordersDao;
  7. }
  8. // 调用dao的方法
  9. // 业务逻辑,写转账业务
  10. public void accountMoney() {
  11. // 小马多1000
  12. ordersDao.addMoney();
  13. // 加入出现异常如下面int i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱
  14. // 解决办法是出现异常后进行事务回滚
  15. int i = 10 / 0;// 事务管理配置后异常已经解决
  16. // 小王 少1000
  17. ordersDao.reduceMoney();
  18. }
  19. }

配置文件:

  1. <!-- 配置c3po连接池 -->
  2. <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
  3. <!-- 注入属性值 -->
  4. <property name="driverClass" value="com.mysql.jdbc.Driver"></property>
  5. <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
  6. <property name="user" value="root"></property>
  7. <property name="password" value="153963"></property>
  8. </bean>
  9. <!-- 第一步:配置事务管理器 -->
  10. <bean id="dataSourceTransactionManager"
  11. class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  12. <!-- 注入dataSource -->
  13. <property name="dataSource" ref="dataSource"></property>
  14. </bean>
  15. <!-- 第二步:配置事务增强 -->
  16. <tx:advice id="txadvice" transaction-manager="dataSourceTransactionManager">
  17. <!-- 做事务操作 -->
  18. <tx:attributes>
  19. <!-- 设置进行事务操作的方法匹配规则 -->
  20. <!-- account开头的所有方法 -->
  21. <!--
  22. propagation:事务传播行为;
  23. isolation:事务隔离级别;
  24. read-only:是否只读;
  25. rollback-for:发生那些异常时回滚
  26. timeout:事务过期时间
  27. -->
  28. <tx:method name="account*" propagation="REQUIRED"
  29. isolation="DEFAULT" read-only="false" rollback-for="" timeout="-1" />
  30. </tx:attributes>
  31. </tx:advice>
  32. <!-- 第三步:配置切面 切面即把增强用在方法的过程 -->
  33. <aop:config>
  34. <!-- 切入点 -->
  35. <aop:pointcut expression="execution(* cn.itcast.service.OrdersService.*(..))"
  36. id="pointcut1" />
  37. <!-- 切面 -->
  38. <aop:advisor advice-ref="txadvice" pointcut-ref="pointcut1" />
  39. </aop:config>
  40. <!-- 对象生成及属性注入 -->
  41. <bean id="ordersService" class="cn.itcast.service.OrdersService">
  42. <property name="ordersDao" ref="ordersDao"></property>
  43. </bean>
  44. <bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
  45. <property name="jdbcTemplate" ref="jdbcTemplate"></property>
  46. </bean>
  47. <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
  48. <property name="dataSource" ref="dataSource"></property>
  49. </bean>

(3)基于注解的方式

OrdersService.java(业务逻辑层)

  1. package cn.itcast.service;
  2. import org.springframework.transaction.annotation.Isolation;
  3. import org.springframework.transaction.annotation.Propagation;
  4. import org.springframework.transaction.annotation.Transactional;
  5. import cn.itcast.dao.OrdersDao;
  6. @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false, timeout = -1)
  7. public class OrdersService {
  8. private OrdersDao ordersDao;
  9. public void setOrdersDao(OrdersDao ordersDao) {
  10. this.ordersDao = ordersDao;
  11. }
  12. // 调用dao的方法
  13. // 业务逻辑,写转账业务
  14. public void accountMoney() {
  15. // 小马多1000
  16. ordersDao.addMoney();
  17. // 加入出现异常如下面int i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱
  18. // 解决办法是出现异常后进行事务回滚
  19. // int i = 10 / 0;// 事务管理配置后异常已经解决
  20. // 小王 少1000
  21. ordersDao.reduceMoney();
  22. }
  23. }

配置文件:

  1. <!-- 配置c3po连接池 -->
  2. <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
  3. <!-- 注入属性值 -->
  4. <property name="driverClass" value="com.mysql.jdbc.Driver"></property>
  5. <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
  6. <property name="user" value="root"></property>
  7. <property name="password" value="153963"></property>
  8. </bean>
  9. <!-- 第一步:配置事务管理器 (和配置文件方式一样)-->
  10. <bean id="dataSourceTransactionManager"
  11. class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
  12. <!-- 注入dataSource -->
  13. <property name="dataSource" ref="dataSource"></property>
  14. </bean>
  15. <!-- 第二步: 开启事务注解 -->
  16. <tx:annotation-driven transaction-manager="dataSourceTransactionManager" />
  17. <!-- 第三步 在方法所在类上加注解 -->
  18. <!-- 对象生成及属性注入 -->
  19. <bean id="ordersService" class="cn.itcast.service.OrdersService">
  20. <property name="ordersDao" ref="ordersDao"></property>
  21. </bean>
  22. <bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
  23. <property name="jdbcTemplate" ref="jdbcTemplate"></property>
  24. </bean>
  25. <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
  26. <property name="dataSource" ref="dataSource"></property>
  27. </bean>