haidaoxianzi 2020-01-01
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中,默认就开启了事务,所以勿须在启动类上添加此注解了.