美文网首页
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