Spring在一个事务中开启另一个事务

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

相关推荐