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中,默認就開啟了事務,所以勿須在啟動類上添加此注解了.