1. 程式人生 > 程式設計 >Spring如何在一個事務中開啟另一個事務

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方法的事務並不會開啟.

解決方法

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中,預設就開啟了事務,所以勿須在啟動類上新增此註解了

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。