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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

若文章对您有帮助,帮忙点个赞!

0
-3
发布时间 2020-01-02 15:01:18
0 条回复(回复会通过微信通知作者)
点击加载更多评论
登录 后再进行评论
(微信扫码即可登录,无需注册)