Spring事务简介
Spring事务作用
- 事务作用:在数据层保障一系列的数据库操作同成功同失败。
- Spring事务作用:在数据层或业务层保障一系列的数据库操作同成功同失败。
需求和分析
- 需求:实现任意两个账户间转账操作。
- 需求微缩:A账户减钱,B账户加钱。
- 分析:
①数据层提供基础操作,指定账户减钱(outMoney),指定账户加钱(inMoney)。
②业务层提供转账操作(transfer),调用减钱与加钱的操作。
③提供2个账号和操作金额执行转账操作。
④基于Spring整合MyBatis环境搭建上述操作。 - 结果分析:
①程序正常执行时,账户金额A减B加,没有问题。
②程序出现异常后,转账失败,但是异常之前操作成功,异常之后操作失败,整体业务失败。
代码实现
环境准备
Spring整合Mybatis相关代码(依赖、JdbcConfig、MybatisConfig、SpringConfig)省略。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
| public interface AccountDao {
@Update("update tbl_account set money = money + #{money} where name = #{name}") void inMoney(@Param("name") String name, @Param("money") Double money);
@Update("update tbl_account set money = money - #{money} where name = #{name}") void outMoney(@Param("name") String name, @Param("money") Double money); }
public interface AccountService {
public void transfer(String out,String in ,Double money) ; }
@Service public class AccountServiceImpl implements AccountService { @Autowired private AccountDao accountDao;
public void transfer(String out,String in ,Double money) { accountDao.outMoney(out,money); int i = 1/0; accountDao.inMoney(in,money); } }
|
在业务层接口上添加Spring事务管理
1 2 3 4 5
| public interface AccountService { @Transactional public void transfer(String out,String in ,Double money) ; }
|
注意事项:
- Spring注解式事务通常添加在业务层接口中而不会添加到业务层实现类中,降低耦合。
- 注解式事务可以添加到业务方法上表示当前方法开启事务,也可以添加到接口上表示当前接口所有方法开启事务。
设置事务管理器(将事务管理器添加到IOC容器中)
说明:可以在JdbcConfig中配置事务管理器。
1 2 3 4 5 6 7
| @Bean public PlatformTransactionManager transactionManager(DataSource dataSource){ DataSourceTransactionManager dtm = new DataSourceTransactionManager(); transactionManager.setDataSource(dataSource); return transactionManager; }
|
注意事项:
- 事务管理器要根据实现技术进行选择。
- MyBatis框架使用的是JDBC事务。
开启注解式事务驱动
1 2 3 4 5 6 7 8
| @Configuration @ComponentScan("edu.heuet") @PropertySource("classpath:jdbc.properties") @Import({JdbcConfig.class,MybatisConfig.class})
@EnableTransactionManagement public class SpringConfig { }
|
运行测试类,查看结果
1 2 3 4 5 6 7 8 9 10 11 12
| @RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = SpringConfig.class) public class AccountServiceTest {
@Autowired private AccountService accountService;
@Test public void testTransfer() throws IOException { accountService.transfer("Tom","Jerry",100D); } }
|
Spring事务角色
- 事务管理员:发起事务方,在Spring中通常指代业务层开启事务的方法。
- 事务协调员:加入事务方,在Spring中通常指代数据层方法,也可以是业务层方法。
Spring事务相关配置
事务配置
说明:对于RuntimeException类型异常或者Error错误,Spring事务能够进行回滚操作。但是对于编译器异常,Spring事务是不进行回滚的,所以需要使用rollbackFor来设置要回滚的异常。
案例:转账业务追加日志
需求和分析
需求:实现任意两个账户间转账操作,并对每次转账操作在数据库进行留痕。
需求微缩:A账户减钱,B账户加钱,数据库记录日志。
分析:
①基于转账操作案例添加日志模块,实现数据库中记录日志。
②业务层转账操作(transfer),调用减钱、加钱与记录日志功能。
实现效果预期:
无论转账操作是否成功,均进行转账操作的日志留痕。
存在的问题:
日志的记录与转账操作隶属同一个事务,同成功同失败。
实现效果预期改进:
无论转账操作是否成功,日志必须保留。
事务传播行为:事务协调员对事务管理员所携带事务的处理态度。
环境整备
1 2 3 4 5 6
| USE spring_db; CREATE TABLE tbl_log( id INT PRIMARY KEY AUTO_INCREMENT, info VARCHAR(255), createDate DATE );
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| public interface LogService { @Transactional void log(String out, String in, Double money); }
@Service public class LogServiceImpl implements LogService {
@Autowired private LogDao logDao; public void log(String out,String in,Double money ) { logDao.log("转账操作由"+out+"到"+in+",金额:"+money); } }
public interface LogDao { @Insert("insert into tbl_log (info,createDate) values(#{info},now())") void log(String info); }
|
在AccountServiceImpl中调用logService中添加日志的方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| @Service public class AccountServiceImpl implements AccountService { @Autowired private AccountDao accountDao;
@Autowired private LogService logService;
public void transfer(String out,String in ,Double money) { try{ accountDao.outMoney(out,money); int i = 1/0; accountDao.inMoney(in,money); }finally { logService.log(out,in,money); } } }
|
在LogService的log()方法上设置事务的传播行为
1 2 3 4 5
| public interface LogService { @Transactional(propagation = Propagation.REQUIRES_NEW) void log(String out, String in, Double money); }
|
运行测试类,查看结果
1 2 3 4 5 6 7 8 9 10 11
| @RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(classes = SpringConfig.class) public class AccountServiceTest { @Autowired private AccountService accountService;
@Test public void testTransfer() throws IOException { accountService.transfer("Tom","Jerry",50D); } }
|
事务传播行为