SpringBoot启动流程分析知识点–@Transactional(二)
一、概述
源码基于SpringBoot 2.7.xx版本
1.1 Spring 中的事务分类
- 🍑编程式事务(手动写代码操作事务)(了解)
- 🍑声明式事务(用注解自动开启和提交事务)(重点)
二、使用详解
2.1 事务传播机制
在使用事务时,如果一个事务方法调用另一个事务方法,那么不同事务之间的执行应该如何调度呢?这就是事务传播的作用,用来调度不同事务执行时的顺序和方式。
MySQL默认的事务传播机制是PROPAGATION_REQUIRED
类型 | 说明 |
---|---|
PROPAGATION_REQUIRED | 如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务 |
PROPAGATION_SUPPORTS | 如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行 |
PROPAGATION_MANDATORY | 如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常 |
PROPAGATION_REQUIRES_NEW | 创建一个新的事务,如果当前存在事务,则把当前事务挂起 |
NOT_SUPPORTED | 以非事务方式运行,如果当前存在事务,则把当前事务挂起 |
NEVER | 以非事务方式运行,如果当前存在事务,则抛出异常 |
NESTED | 如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行 |
1.PROPAGATION_REQUIRED=0:
如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
无父事务时:子事务作为独立事务执行
有父事务时:子事务中的操作串入父事务中执行,并且一起提交,一个操作失败全部回滚
2.PROPAGATION_SUPPORTS=1:
如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
无父事务时:以非事务方式执行
有父事务时:加入父事务执行,等同于PROPAGATION_REQUIRED
3.PROPAGATION_MANDATORY=2:
如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。
无父事务时:抛出异常
有父事务时:加入父事务执行,等同于PROPAGATION_REQUIRED
4.PROPAGATION_REQUIRES_NEW=3:
创建一个新的事务,如果当前存在事务,则把当前事务挂起。
挂起(Suspend):通知TransactionManager不再检查某事务的状态,直到Resume
AbstractPlatformTransactionManager. handleExistingTransaction()
无父事务时:子事务新建事务作为独立事务执行
有父事务时:子事务新建事务作为独立事务执行,独立提交;
T1{ O(A); T2{ O(B); O(C); }; O(D);
};
如此示例:
子事务T2不受父事务T1回滚的影响,但仍作为T1的子逻辑,
O(D)失败,O(A)回滚,T2中的事务不回滚;
T2失败回滚,T1捕获异常后,可以选择提交或回滚,未捕获异常,同T2一起回滚。
5.NOT_SUPPORTED=4:
以非事务方式运行,如果当前存在事务,则把当前事务挂起。
无父事务时:以非事务方式执行
有父事务时:挂起父事务,自己按照无事务方式运行
子事务自身无回滚,出现异常若向上抛,可能导致父事务回滚
父事务回滚时,不会影响子事务。
6.NEVER=5:
以非事务方式运行,如果当前存在事务,则抛出异常。
无父事务时:以非事务方式执行
有父事务时:抛出异常(若不处理会导致父事务回滚)
7.NESTED=6:
如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行
无父事务时:创建独立事务,等同于PROPAGATION_REQUIRED
有父事务时:嵌套在父事务之内
子事务依赖父事务:子事务于父事务提交时提交;父事务回滚,子事务也回滚。
Savepoint:子事务回滚时,父事务不回滚
2.2 示例
- 编程式事务(手动写代码操作事务)(了解)
@Slf4j
@RestController
@RequestMapping("/trans")
public class TransactionController {
@Autowired
private UserService userService;
//1、拿到事务管理器
@Autowired
private DataSourceTransactionManager dataSourceTransactionManager;
//2、定义事务属性
@Autowired
private TransactionDefinition transactionDefinition;
@RequestMapping("/addUser")
public Integer addUser(String username, String password) {
//3、开启事务
TransactionStatus transaction = dataSourceTransactionManager.getTransaction(transactionDefinition);
//4、插入数据
User user = new User(username, password);
Integer result = userService.insert(user);
log.info("影响的行数是:" + result);
//5、提交事务
dataSourceTransactionManager.commit(transaction);
//6、回滚事务
// dataSourceTransactionManager.rollback(transaction);
return result;
}
}
- 声明式事务(用注解自动开启和提交事务)(重点)
/**
* 自动提交事务
*/
@Slf4j
@RestController
@RequestMapping("/trans")
public class TransactionController {
@Autowired
private UserService userService;
// 声明式事务的实现很简单,只需要在需要的⽅法上添加 @Transactional 注解就可以实现了,⽆需⼿动
// 开启事务和提交事务,进⼊⽅法时⾃动开启事务,⽅法执⾏完会⾃动提交事务;如果中途发⽣了没有处
// 理的异常会⾃动回滚事务
@RequestMapping("/addUser2")
@Transactional
public Integer addUser2(String username, String password) {
// 插入数据
User user = new User(username, password);
Integer result = userService.insert(user);
// 模拟异常
int a = 10 / 0;
return result;
}
}