美文网首页
SpringBoot中@Transactional自调用失效问题

SpringBoot中@Transactional自调用失效问题

作者: 蓝Renly | 来源:发表于2019-04-28 23:37 被阅读0次

1.关于事务

项目类型 脏读 不可重复读 幻读
未提交读
读写提交 ×
可重复读 × ×
串行化 × × ×

2.传播行为

传播行为是方法之间调用事务采取的策略问题.

例如:执行一个批量程序,它会处理很多交易,绝大多数交易可以顺利完成,极少数会发生异常,这时不应该回滚所有交易.

传播行为枚举
package org.springframe.transaction.annotation
public enum Prooagation{
    /**
    * 重点!!!
    *需要事务,它是默认传播行为,如果当前存在事务,就沿用当前事务
    *否则新建一个事务运行子方法
    */
    REQUIRED(TransactionDefinition.PROPAGATION_REQUIRED)
    
    /**
    * 重要!!!
    *无论当前事务是否存在,都会创建新事物运行方法,
    *这样新事物就可以拥有新的锁和隔离级别等特性,与当前事务相互独立.
    */
    REQUIRES_NEW(TransactionDefinition.REQUIRES_NEW)
    
    /**
    * 重要!!!
    *在当前方法调用子方法时,如果字方法发生异常,
    *只回滚子方法执行过的SQL,而不回滚当前方法的事务.
    */
    NESTTED(TransactionDefinition.REQUIRES_NESTED)
    
    /**
    *这样新事物就可以拥有新的锁和隔离级别等特性,与当前事务相互独立.
    */
    REQUIRES_NEW(TransactionDefinition.REQUIRES_NEW)
    
        /**
    *支持事务,如果当前存在事务,就沿用当前事务,
    *如果不存在,则继续采用无事务的方式运行子方法
    */
    SUPPORTS(TransactionDefinition.PROPAGATION_SUPPORTS)
    
    /**
    *必须使用事务,如果当前没有事务,则会抛出异常,
    *这样新事物就可以拥有新的锁和隔离级别等特性,与当前事务相互独立.
    */
    MANDATORY(TransactionDefinition.PROPAGATION_MANDATORY)
    
    .............................
    .............................
}

测试

@Service
public class UserBatchServiceImpl implents UserBatchService{
    @Autowired
    private UserService userService = null;
    
    @Override
    @Transactional(isolation = Isolation.READ_COMMITED,propagation = Propagation.REQUIRED)
    public int  insertUsers(List<User> userLists){
        int count = 0;
        for (User user : userList){
            count += userService.insertUser(user)
        }
        return count;
    }
}

注意:

NESTED:传播行为会沿用当前事务的隔离级别和锁等特性.

REQUIRES_NEW:可以拥有自己独立的隔离级别和 锁等特性.在应用中需要注意.

3.Transactional自调用失效

Spring数据库事务约定的实现原理是AOP,而AOP原理是动态代理,在自调用过程中,是类自身的调用,而不是代理对象的调用,就不会产生AOP.

那么用一个Service去调用另一个Service,这样就是代理对象调用.(如:用批量插入UserBatchService去调用单个插入UserService);

也可以通过从Spring IoC中取出代理对象,克服自调用问题:

// 1.实现ApplicationContextAware
private AplicationContext applicationContext = null;
// 2.从IoC容器中取出代理对象
UserService useService = applicationContext.getBean(UserService.class) // 当前类的实现接口
// 3.设置隔离级别和传播行为.

从IoC容器取出的对象是一个代理对象,通过它克服自调用的问题.

相关文章

网友评论

      本文标题:SpringBoot中@Transactional自调用失效问题

      本文链接:https://www.haomeiwen.com/subject/ktisnqtx.html