Spring事务不能回滚的深层次原因

简介: Spring事务不能回滚的深层次原因

开头总述

Spring在同一个类中调用function,事务会失效。

Spring事务是基于AOP代理来实现的。而AOP是使用JDK动态代理来实现的。

第一次试验

    /**
     * 父类调用子类
     * 子类失败,不能影响父类
     * 
     * 预期效果:child回滚,parent插入成功
     * 第一次试验 真实效果:都插入成功,child方法因为try catch导致事务未起作用。
     */  
    @Override
    @Transactional
    public void insertParent(){
        try {
            insertChild();
        } catch (Exception e) {
            // TODO: handle exception
        }
        user.setName("parent_5");
        user.setAge((byte)1);
        userMapper.insertSelective(user);
    }
    @Override
    @Transactional(propagation=Propagation.REQUIRES_NEW)
    public void insertChild(){
        user.setName("child_5");
        user.setAge((byte)1);
        userMapper.insertSelective(user);
        int i=1/0;
    }

执行结果

原因分析

   child方法被try catch住了,同时事务未生效。

  在SpringIoC容器中返回的调用的对象是代理对象而不是真实的对象,只有被动态代理直接调用的才会产生事务。this调用并非代理对象。this.insertChild走的是真实对象(真实对象)。导致child上的事务无效,先执行insert语句,执行成功以后才抛出异常。异常被parent中的try catch捕获到。不影响parent方法的执行。

第二次试验

 

   /**
    * 预期效果:child回滚,parent插入成功
    * 第二试验 真实效果:child插入成功,parent插入回滚
    */
    @Override
    @Transactional
    public void insertParent(){        
        user.setName("parent_7");
        user.setAge((byte)1);
        userMapper.insertSelective(user);
        insertChild();        
    }
    @Override
    @Transactional(propagation=Propagation.REQUIRES_NEW)
    public void insertChild(){
        user.setName("child_7");
        user.setAge((byte)1);
        userMapper.insertSelective(user);
        int i=1/0;
    }

执行结果

   

原因分析

   去掉try catch,child方法抛出异常,导致回滚。同时异常向上抛,导致parent方法也回滚。

   child方法的事务失效,相当于child方法中的code挪到了parent方法中,也类似于在parent中抛出异常。

解决方案

   AOP需要先暴露出来。

<!-- 开启切面-->

<!-- 暴露AOP代理到ThreadLocal -->

<aop:aspectj-autoproxy expose-proxy="true"></aop:aspectj-autoproxy>

获取AOP代理需要从AOP的上下文来获取。得到当前AopProxy,然后通过AOP代理来调用child方法。

 

   /**
    * 预期效果:child回滚,parent插入成功
    * 第三试验真实效果:child回滚,parent插入成功
    */   
    @Override
    @Transactional
    public void insertParent(){        
        try {
            /**
             * AopContext.currentProxy()就类似于JDK代理中的
             * Proxy.newInstance得到的Proxy对象
             * UserService proxy=(UserService) AopContext.currentProxy();
             * proxy.insertChild();
             */
            ((UserService)AopContext.currentProxy()).insertChild();
        } catch (Exception e) {
            // TODO: handle exception
        }        
        user.setName("parent_7");
        user.setAge((byte)1);
        userMapper.insertSelective(user);    
    }
    @Override
    @Transactional(propagation=Propagation.REQUIRES_NEW)
    public void insertChild(){
        user.setName("child_7");
        user.setAge((byte)1);
        userMapper.insertSelective(user);
        int i=1/0;
    }

执行结果

论:非PROPAGATION_NESTED的标注的嵌套事务不使用动态代理的方式来调用会导致子方法里的事务失效


相关文章
|
3月前
|
安全 Java 数据库
一天十道Java面试题----第四天(线程池复用的原理------>spring事务的实现方式原理以及隔离级别)
这篇文章是关于Java面试题的笔记,涵盖了线程池复用原理、Spring框架基础、AOP和IOC概念、Bean生命周期和作用域、单例Bean的线程安全性、Spring中使用的设计模式、以及Spring事务的实现方式和隔离级别等知识点。
|
5天前
|
Java 开发者 Spring
Spring高手之路24——事务类型及传播行为实战指南
本篇文章深入探讨了Spring中的事务管理,特别是事务传播行为(如REQUIRES_NEW和NESTED)的应用与区别。通过详实的示例和优化的时序图,全面解析如何在实际项目中使用这些高级事务控制技巧,以提升开发者的Spring事务管理能力。
15 1
Spring高手之路24——事务类型及传播行为实战指南
|
2月前
|
Java 数据库连接 数据库
spring复习05,spring整合mybatis,声明式事务
这篇文章详细介绍了如何在Spring框架中整合MyBatis以及如何配置声明式事务。主要内容包括:在Maven项目中添加依赖、创建实体类和Mapper接口、配置MyBatis核心配置文件和映射文件、配置数据源、创建sqlSessionFactory和sqlSessionTemplate、实现Mapper接口、配置声明式事务以及测试使用。此外,还解释了声明式事务的传播行为、隔离级别、只读提示和事务超时期间等概念。
spring复习05,spring整合mybatis,声明式事务
|
2月前
|
Java 测试技术 数据库
Spring事务传播机制(最全示例)
在使用Spring框架进行开发时,`service`层的方法通常带有事务。本文详细探讨了Spring事务在多个方法间的传播机制,主要包括7种传播类型:`REQUIRED`、`SUPPORTS`、`MANDATORY`、`REQUIRES_NEW`、`NOT_SUPPORTED`、`NEVER` 和 `NESTED`。通过示例代码和数据库插入测试,逐一展示了每种类型的运作方式。例如,`REQUIRED`表示如果当前存在事务则加入该事务,否则创建新事务;`SUPPORTS`表示如果当前存在事务则加入,否则以非事务方式执行;`MANDATORY`表示必须在现有事务中运行,否则抛出异常;
121 4
Spring事务传播机制(最全示例)
|
1月前
|
Java 关系型数据库 MySQL
Spring事务失效,我总结了这7个主要原因
本文详细探讨了Spring事务在日常开发中常见的七个失效原因,包括数据库不支持事务、类不受Spring管理、事务方法非public、异常被捕获、`rollbackFor`属性配置错误、方法内部调用事务方法及事务传播属性使用不当。通过具体示例和源码分析,帮助开发者更好地理解和应用Spring事务机制,避免线上事故。适合所有使用Spring进行业务开发的工程师参考。
28 2
|
1月前
|
Java 程序员 Spring
Spring事务的1道面试题
每次聊起Spring事务,好像很熟悉,又好像很陌生。本篇通过一道面试题和一些实践,来拆解几个Spring事务的常见坑点。
Spring事务的1道面试题
|
2月前
|
Java Spring
Spring 事务传播机制是什么?
Spring 事务传播机制是什么?
22 4
|
1月前
|
监控 Java 数据库
Spring事务中的@Transactional注解剖析
通过上述分析,可以看到 `@Transactional`注解在Spring框架中扮演着关键角色,它简化了事务管理的复杂度,让开发者能够更加专注于业务逻辑本身。合理运用并理解其背后的机制,对于构建稳定、高效的Java企业应用至关重要。
43 0
|
3月前
|
XML Java 数据库
Spring5入门到实战------15、事务操作---概念--场景---声明式事务管理---事务参数--注解方式---xml方式
这篇文章是Spring5框架的实战教程,详细介绍了事务的概念、ACID特性、事务操作的场景,并通过实际的银行转账示例,演示了Spring框架中声明式事务管理的实现,包括使用注解和XML配置两种方式,以及如何配置事务参数来控制事务的行为。
Spring5入门到实战------15、事务操作---概念--场景---声明式事务管理---事务参数--注解方式---xml方式
|
3月前
|
前端开发 Java 数据库连接
一天十道Java面试题----第五天(spring的事务传播机制------>mybatis的优缺点)
这篇文章总结了Java面试中的十个问题,包括Spring事务传播机制、Spring事务失效条件、Bean自动装配方式、Spring、Spring MVC和Spring Boot的区别、Spring MVC的工作流程和主要组件、Spring Boot的自动配置原理和Starter概念、嵌入式服务器的使用原因,以及MyBatis的优缺点。
下一篇
无影云桌面