解决方法A调用方法B的事务控制问题

 更新时间:2023年07月26日 09:13:54   作者:绘码先生  
这篇文章主要介绍了解决方法A调用方法B的事务控制问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

关于方法A调用方法B的事务控制问题

在这里插入图片描述

实测,不管是上图还是下图,得到的结论都是一致的,那就是:

前提是同一个类里的方法调用,controller层类调用service的B方法,事务自然会生效的

a.异常发生后,被调用方法(B)是否添加事务控制,对于事务的回滚是否并不产生影响

b.调用B的方法,添加了事务控制才能实现异常事务回滚,不管被调用方法B是否有事务控制

c.不管嵌套调用了多少个方法,只要最顶层方法(没被同一个类中其他方法调用的且调用了同一个类里其他方法的方法),有事务控制,那么不管是哪个被调用的方法异常,整个调用的业务数据都回滚,不管被调用的方法是否有添加了事务的控制

在这里插入图片描述

其实,最好的办法是根据自己实际调用的情况,模拟测试一下就知道了。

同一个类的不同方法,A方法没有@Transactional,B方法有@Transactional,A调用B方法,事务不起作用

问题

同一个类的不同方法,A方法没有@Transactional,B方法有@Transactional,A调用B方法,事务不起作用

原理解析

spring 在扫描bean的时候会扫描方法上是否包含@Transactional注解,如果包含,spring会为这个bean动态地生成一个子类(即代理类,proxy),代理类是继承原来那个bean的。

此时,当这个有注解的方法被调用的时候,实际上是由代理类来调用的,代理类在调用之前就会启动transaction。

然而,如果这个有注解的方法是被同一个类中的其他方法调用的,那么该方法的调用并没有通过代理类,而是直接通过原来的那个bean,所以就不会启动transaction,我们看到的现象就是@Transactional注解无效。

    //接口
    interface Service {
        void A();
        void B();
    }
    //目标类,实现接口
    class ServiceImpl implements Service {
        //no annotation here
        @Override
        public void A() {
            this.B();
        }
        @Transactional
        @Override
        public void B() {
            System.out.println("execute doNeedTx in ServiceImpl");
        }
    }
    //代理类,也要实现相同的接口
    class ProxyByJdkDynamic implements Service {
        //包含目标对象
        private Service target;
        public ProxyByJdkDynamic(Service target) {
            this.target = target;
        }
        //目标类中此方法带注解,进行特殊处理
        @Override
        public void B() {
            //开启事务
            System.out.println("-> create Tx here in Proxy");
            //调用目标对象的方法,该方法已在事务中了
            target.B();
            //提交事务
            System.out.println("<- commit Tx here in Proxy");
        }
        //目标类中此方法没有注解,只做简单的调用
        @Override
        public void A() {
            //直接调用目标对象方法
            target.A();
        }
    }

那回到一开始的问题,我们调用的方法A不带注解,因此代理类不开事务,而是直接调用目标对象的方法。

当进入目标对象的方法后,执行的上下文已经变成目标对象本身了,因为目标对象的代码是我们自己写的,和事务没有半毛钱关系,此时你再调用带注解的方法,照样没有事务,只是一个普通的方法调用而已。

简单来说,内部调用本类方法,不会再走代理了,所以B的事务不起作用

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 一文了解Java读写锁ReentrantReadWriteLock的使用

    一文了解Java读写锁ReentrantReadWriteLock的使用

    ReentrantReadWriteLock称为读写锁,它提供一个读锁,支持多个线程共享同一把锁。这篇文章主要讲解一下ReentrantReadWriteLock的使用和应用场景,感兴趣的可以了解一下
    2022-10-10
  • FastJson时间格式化问题避坑经验分享

    FastJson时间格式化问题避坑经验分享

    这篇文章主要为大家介绍了FastJson时间格式化问题避坑经验分享,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • Spring框架AOP面向切面编程原理全面分析

    Spring框架AOP面向切面编程原理全面分析

    这篇文章主要介绍了Spring框架AOP面向切面编程的全面分析,文中附含详细的示例代码分析,有需要的朋友可以借鉴参考下,希望能够有所帮助
    2021-09-09
  • Spring Boot动态加载Jar包与动态配置实现

    Spring Boot动态加载Jar包与动态配置实现

    随着项目的不断演进和业务需求的增长,很多场景下需要实现系统的动态性和灵活性,本文主要介绍了Spring Boot动态加载Jar包与动态配置实现,具有一定的参考价值,感兴趣的可以了解一下
    2024-02-02
  • java实现简单TCP聊天程序

    java实现简单TCP聊天程序

    这篇文章主要为大家详细介绍了java实现简单TCP聊天程,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • 详解SpringBoot简化配置分析总结

    详解SpringBoot简化配置分析总结

    这篇文章主要介绍了详解SpringBoot简化配置分析总结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-10-10
  • java使用poi生成excel的步骤

    java使用poi生成excel的步骤

    2010以上格式使用XSSFWorkBook对象, 2003格式使用HSSFWorkBook对象, 其他对象操作基本一样,本文重点给大家介绍java使用poi生成excel的相关知识,感兴趣的朋友一起看看吧
    2022-04-04
  • springmvc组件中的HandlerMapping解析

    springmvc组件中的HandlerMapping解析

    这篇文章主要介绍了springmvc九大组件中的HandlerMapping解析,HandlerMapping表示的是一个URL与一个Handler(可以简单的理解为Controller中有@RequestMapping注解的方法)之间的映射关系,需要的朋友可以参考下
    2023-09-09
  • Java IO之File 类详解

    Java IO之File 类详解

    这篇文章主要为大家介绍了vue组件通信的几种方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-01-01
  • java 将数据加载到内存中的操作

    java 将数据加载到内存中的操作

    这篇文章主要介绍了java 将数据加载到内存中的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-09-09

最新评论