• 同类方法调用 @Transactional 失效与解决


    一、理解@Transactional底层的AOP原理


    最初调用的是AOP代理对象而不是目标对象。执行事务切面,事务切面内部通过TransactionInterceptor环绕增强进行事务的增强,即进入目标方法之前开启事务,退出目标方法时提交/回滚事务。

    public interface AService {
        public void a();
        public void b();
    }
    
    • 1
    • 2
    • 3
    • 4
    @Service()
    public class AServiceImpl implements AService{
        @Transactional(propagation = Propagation.REQUIRED)
        public void a() {
            this.b();
        }
        @Transactional(propagation = Propagation.REQUIRES_NEW)
        public void b() {
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    二、同类方法的调用无法实现切面中的增强

    在这里插入图片描述

    此处this指向目标对象,因此调用this.b()将不会执行b事务切面,即不会执行事务增强,因此b方法的事务定义“@Transactional(propagation = Propagation.REQUIRES_NEW)”将不会实施。即结果是b和a的事务定义是一样的(可以看到事务切面只对a方法进行了事务增强,没有对b方法进行增强)。

    1. b中的事务会不会生效?
      不会,a的事务会生效,b中不会有事务,因为a中调用b属于内部调用,没有通过代理,所以不会有事务产生。
    2. 如果想要b中有事务存在,要如何做?
      此处a方法中调用b方法时,只要通过AOP代理调用b方法即可执行事务切面,进行事务增强。

    三、暴露 AOP 代理

    1. 常规项目
      设置expose-proxy属性为true
    2. SpringBoot项目,在启动类上加@EnableAspectJAutoProxy(exposeProxy = true)就可以了。

    更改后:
    使用(AService)AopContext.currentProxy();获取当前代理,将this.b()改为((AService)AopContext.currentProxy()).b()

    四、三种判断AOP代理对象的方法

    1. AopUtils.isAopProxy(bean):是否是代理对象。
    2. AopUtils.isJdkDynamicProxy(bean):是否是JDK动态代理对象。
    3. AopUtils.isCglibProxy(bean):是否是CGLIB动态代理对象。
  • 相关阅读:
    LeetCode_栈_困难_1106.解析布尔表达式
    spring实现基于注解的@Bean配置原理
    Kafka - 3.x 文件存储不完全指北
    Redis缓存击穿及解决问题
    C和指针 第10章 结构和联合 10.1 结构基础知识
    BD就业复习第三天
    数据分析--数据预处理
    直流电源供电 LED升压 恒流驱动IC 方案AP9193
    中缀表达式 - 栈实现综合计算器
    Python潮流周刊#1:如何系统地自学Python?
  • 原文地址:https://blog.csdn.net/ChineseSoftware/article/details/127602310