0
点赞
收藏
分享

微信扫一扫

Spring Tx (八) (SpringBoot注解@Transactional 2)


  1. @Transactional失败的情况
  2. @Transactional的源码分析
  3. @Transactional失败的情况以及对应的源码分析

文章目录

  • ​​1.@Transactional 失败的情况​​
  • ​​1.1 类内部访问​​
  • ​​1.2 私有方法​​
  • ​​1.3 异常不匹配​​
  • ​​1.4 多线程​​
  • ​​2.@Transactional 源码分析​​
  • ​​2.1 @Transactional 执行机制​​
  • ​​3.@Transactional 失效原因​​
  • ​​3.1 private 导致事务不生效原因​​
  • ​​3.2 异常不匹配原因​​

1.@Transactional 失败的情况

链接: ​​javascript:void(0)​​

UserDao

public interface UserDao {
// select * from user_test where uid = "#{uid}"
public MyUser selectUserById(Integer uid);
// update user_test set uname =#{uname},usex = #{usex} where uid = #{uid}
public int updateUser(MyUser user);
}

UserController

@Service
public class UserController {
@Autowired
private UserDao userDao;

public void update(Integer id) {
MyUser user = new MyUser();
user.setUid(id);
user.setUname("张三-testing");
user.setUsex("女");
userDao.updateUser(user);
}

public MyUser query(Integer id) {
MyUser user = userDao.selectUserById(id);
return user;
}

// 正常情况
@Transactional(rollbackFor = Exception.class)
public void testSuccess() throws Exception {
Integer id = 1;
MyUser user = query(id);
System.out.println("原记录:" + user);
update(id);
throw new Exception("事务生效");
}
}

@Transactional 事务不生效的几种情况

Spring Tx (八) (SpringBoot注解@Transactional 2)_子线程

  1. 类内部访问:A 类的 a1 方法没有标注 @Transactional,a2 方法标注 @Transactional,在 a1 里面调用 a2;
  2. 私有方法:将 @Transactional 注解标注在非 public 方法上;
  3. 异常不匹配:@Transactional 未设置 rollbackFor 属性,方法返回 Exception 等异常;
  4. 多线程:主线程和子线程的调用,线程抛出异常。

1.1 类内部访问

UserController.testInteralCall()

public void testInteralCall() throws Exception {
testSuccess();
throw new Exception("事务不生效:类内部访问");
}

这里 testInteralCall() 没有标注 @Transactional

public static void main(String[] args) throws Exception {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserController uc = (UserController) applicationContext.getBean("userController");
try {
uc.testSuccess();
} finally {
MyUser user = uc.query(1);
System.out.println("修改后的记录:" + user);
}
}

// 原记录:MyUser(uid=1, uname=张三, usex=女)
// 修改后的记录:MyUser(uid=1, uname=张三-testing, usex=女)

@Transactional 的工作机制是基于 AOP 实现, Aop 是动态代理, 通过代理调用testSuccess(), 通过Aop增强, 增强的逻辑其实就是在 testSuccess() 的前后分别加上开启、提交事务的逻辑。

通过 testInteralCall() 去调用 testSuccess(), testSuccess() 前后不会进行任何增强操作, ​​类内部调用, 不会进行代理方式访问​

获取本对象的代理对象,再进行调用。

@Service
public class OrderService {

public void insert() {
OrderService proxy = (OrderService) AopContext.currentProxy();
proxy.insertOrder();
}

@Transactional
public void insertOrder() {
//SQL操作
}
}

1.2 私有方法

在私有方法上,添加 @Transactional 注解也不会生效

@Transactional(rollbackFor = Exception.class)
private void testPirvateMethod() throws Exception {
Integer id = 1;
MyUser user = query(id);
System.out.println("原记录:" + user);
update(id);
throw new Exception("测试事务生效");
}

1.3 异常不匹配

@Transactional 没有设置 rollbackFor = Exception.class

@Transactional
public void testExceptionNotMatch() throws Exception {
Integer id = 1;
MyUser user = query(id);
System.out.println("原记录:" + user);
update(id);
throw new Exception("事务不生效:异常不匹配");
}

@Transactional 注解默认处理运行时异常, RuntimeException.class

1.4 多线程

父线程抛出异常

父线程抛出异常,子线程不抛出异常

public void testSuccess() throws Exception {
Integer id = 1;
MyUser user = query(id);
System.out.println("原记录:" + user);
update(id);
}
@Transactional(rollbackFor = Exception.class)
public void testMultThread() throws Exception {
new Thread(new Runnable() {
@SneakyThrows
@Override
public void run() {
testSuccess();
}
}).start();
throw new Exception("测试事务不生效");
}

父线程抛出线程, 事务回滚, 因为子线程是单独存在的, 和父线程不在同一个事务, 所以子线程的修改不会回滚。

子线程抛出异常

父线程不抛出异常,子线程抛出异常

ublic void testSuccess() throws Exception {
Integer id = 1;
MyUser user = query(id);
System.out.println("原记录:" + user);
update(id);
throw new Exception("测试事务不生效");
}
@Transactional(rollbackFor = Exception.class)
public void testMultThread() throws Exception {
new Thread(new Runnable() {
@SneakyThrows
@Override
public void run() {
testSuccess();
}
}).start();
}

由于子线程的异常不会被外部的线程捕获,所以父线程不抛异常,事务回滚没有生效。

2.@Transactional 源码分析

2.1 @Transactional 执行机制

Spring Tx (八) (SpringBoot注解@Transactional 2)_spring boot_02

interceptorOrInterceptionAdvice 是 TransactionInterceptor的实例

this 是 ReflectiveMethodInvocation 对象,成员对象包含 UserController 类、testSuccess() 方法、入参和代理对象等。

Spring Tx (八) (SpringBoot注解@Transactional 2)_子线程_03

invoke():

Spring Tx (八) (SpringBoot注解@Transactional 2)_spring boot_04

事务的核心逻辑: 事务是否开启、目标方法执行、事务回滚、事务提交。

Spring Tx (八) (SpringBoot注解@Transactional 2)_java_05

3.@Transactional 失效原因

3.1 private 导致事务不生效原因

getTransactionAttribute(): 获取txAttr 变量, 读取@Transactional 的配置, 如果这个txAttr = null, 不会走后面逻辑

Spring Tx (八) (SpringBoot注解@Transactional 2)_抛出异常_06

getTransactionAttribute()

Spring Tx (八) (SpringBoot注解@Transactional 2)_spring_07

Spring Tx (八) (SpringBoot注解@Transactional 2)_抛出异常_08

allowPublicMethodsOnly() 一直返回 false, isPublic() 判断是否是 Public

Spring Tx (八) (SpringBoot注解@Transactional 2)_spring_09

3.2 异常不匹配原因

Spring Tx (八) (SpringBoot注解@Transactional 2)_子线程_10

rollbackOn(): 判断该异常是否能进行回滚, 这个需要判断主方法抛出的 Exception() 异常,是否在 @Transactional 的配置中。

Spring Tx (八) (SpringBoot注解@Transactional 2)_java_11

getDepth(): 异常规则匹配逻辑

Spring Tx (八) (SpringBoot注解@Transactional 2)_spring boot_12

如果没有为 null, 那么会走默认的异常捕获方式。

Spring Tx (八) (SpringBoot注解@Transactional 2)_子线程_13

Spring Tx (八) (SpringBoot注解@Transactional 2)_java_14

默认只对 RuntimeException 和 Error 的异常执行回滚。


举报

相关推荐

0 条评论