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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM