这篇文章主要介绍了Spring如何在一个事务中开启另一个事务,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
spring使用@Transactional开启事务,而且该注解使用propagation属性来指定事务的传播级别
@Transactional(propagation =Propagation.REQUIRES_NEW) // 开启一个新事务
使用REQUIRES_NEW就会开启一个新的事务吗? 答案并不是.
请看下面的这个示例
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
import qinfeng.zheng.learnpagequery.domain.UserDO;
import qinfeng.zheng.learnpagequery.mapper.UserMapper;
@Service
public class UserService {
@Autowired
private UserMapper userMapper;
@Transactional(rollbackFor = Exception.class)
public void doSomething(UserDO userDo) {
insert(userDo);
doOther();
}
@Transactional(propagation =Propagation.REQUIRES_NEW) // 开启一个新事务
public void insert(UserDO userDo) {
userMapper.insert(userDo);
}
public void doOther() {
System.out.println("做一些其它的事,比如调用其它的系统");
}
}
在调用doSomething方法时,开启了一个事务,该方法中包括insert和doOther, 但是insert方法上也开启了一个事务. 按道理应该有两个事务控制,可事实上并不是, insert方法的事务无效. 这就跟spring事务原理有关系, spring框架是通过TransactionInterceptor类来控制事务开启,提交,回滚等, 它会创建一个目标类的代理类. 而在本示例中,doSomething方法调用insert方法时,并不是通过代理类去调用,而是通过this调用本身的方法insert方法.所以insert方法的事务并不会开启.
解决方法
1. 将insert方法抽取到另一个XxxService方法中, 然后再将这个XxxService注入到UserService类中,通过xxxService.insert()调用, 这样insert方法的事务就会生效了.
2. 第2种方式通过AopContext创建一个代理
在项目启动类上开启 exposeProxy = true
@EnableAspectJAutoProxy(proxyTargetClass = true, exposeProxy = true)
@Transactional(rollbackFor = Exception.class)
public void doSomething(UserDO userDo) {
UserService userService = (UserService) AopContext.currentProxy();
userService.insert(userDo); // 这样insert方法事务生效
doOther();
}
备注: 在springboot1.x中使用@EnableTransactionManagement开启事务, 但是在springboot2.x中,默认就开启了事务,所以勿须在启动类上添加此注解了