事务
事务就是为了保证多次数据库操作的原子性。举个简单的例子
买商品第一步要扣钱,第二步要扣库存。如果没有事务,一旦第一步与第二步之间出现了异常,那么钱是扣了,库存却没变,这显然不符合业务场景。要么都成功要嘛都失败
-
在
springboot
中使用事务就很简单了,首先引入依赖spring-tx
,但是mybatis-plus
的依赖中已经引入,因此又少了一步 -
开启事务,在
Springboot
的启动类,或者某个@Configuration
的类上加上@EnableTransactionManagement
开启事务。因为这是数据库相关,所以我加在了mybatis-plus
的配置类上 - 只要在需要使用事务的方法上加上
@Transactional
就可以开启事务了,还是很简单的
注意点
1. @Transactional
默认回滚的是RuntimeException
也就是说如果抛出的不是RuntimeException
的异常,数据库是不会回滚的。但是所幸的是,在spring框架下,所有的异常都被org.springframework
重写为RuntimeException
因此不需要太担心
@Transactional
public void buy() throws Exception {
1. 扣钱
throw new 非RuntimeException异常("发生异常");
2. 扣库存
}
2. 还有如果在异常发生时,程序员自己手动捕获处理了,异常也不会回滚
@Transactional
public void buy() throws Exception {
try{
1. 扣钱
} catch (Exception e) {
catch了自己处理,也就是异常被自己吞了,外层并不知道,此时也不会回滚
}
3. 扣库存
}
正因为如此,如果你回去看一下我们自己写的APIException
,它就继承了RuntimeException
,因此,如果你使用的是我们自己定义的异常,你只管直接抛异常就行,回滚、日志、统一异常都已经处理好了
// 有一种处理方法是指定回滚的异常
@Transactional(rollbackFor = Exception.class)
public void buy() throws Exception {
1. 扣钱
// 如果此时有一些业务异常
throw new APIException("钱太少拉!不够扣!");
2. 扣库存
}