Spring如何在一个事务中开启另一个事务
这篇文章主要介绍了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 方法的事务并不会开启。
解决方法
-
将 insert 方法抽取到另一个 XxxService 方法中, 然后再将这个 XxxService 注入到 UserService 类中,通过xxxService.insert() 调用,这样 insert 方法的事务就会生效了.
-
第2种方式通过 AopContext 创建一个代理
在项目启动类上开启 exposeProxy = true
@EnableAspectJAutoProxy(proxyTargetClass = true, exposeProxy = true)
在业务代码中调用的地方,通过AopContext.currentProxy()
来调用方法就可以了
@Transactional(rollbackFor = Exception.class)
public void doSomething(UserDO userDo) {
UserService userService = (UserService) AopContext.currentProxy();
userService.insert(userDo); // 这样insert方法事务生效
doOther();
}
备注: 在 springboot1.x 中使用@EnableTransactionManagement
开启事务, 但是在 springboot2.x 中,默认就开启了事务,所以勿须在启动类上添加此注解了。