同一个类中调用方法spring事务注解@Transactional如何生效?

要解答这个问题,要分两步走:

  • 同一个类中,方法A调用方法B,为什么会导致@Transactional事务失效
  • 如何避免事务失效

事务失效原因

我们先来看一个正常的依赖注入。

InjectTest注入了MyServiceImpl,可以调用MyServiceImpl#doSomething()。

但是,当我们给MyServiceImpl#doSomething()加上@Transactional注解后:

myServiceImpl的对象变成了MyServiceImpl <math xmlns="http://www.w3.org/1998/Math/MathML"> E n h a n c e r B y S p r i n g C G L I B EnhancerBySpringCGLIB </math>EnhancerBySpringCGLIB9f78d9e5@5810,很明显是Spring通过CGLIB代理出来的对象。也就是说,InjectTest此时注入的是代理对象,而代理对象内部持有target目标对象。

本次方法调用最终会到达目标对象,也就是真正的MyServiceImpl:

整个过程类似这样:

左边代理对象,右边目标对象,调用代理对象,最终会委托到目标对象。

回到我们的问题:同一类中A方法调用B方法为什么会导致事务失效呢?

因为在add方法中通过this.sub()调用时,本质上根本不会走代理对象的逻辑,而目标对象内部并不存在事务相关的增强逻辑。

解决办法

知道原因后,就很好解决了。最简单的方法是使用AopContext获取实际的代理对象。

那么,怎么在目标对象中获取到实际的代理对象proxy呢?

Spring提供了AopContext。

开启exposeProxty=true以后,Spring会在AopContext中共享当前代理对象(本质是在本次请求中通过ThreadLocal共享代理对象)。

具体代码是:

开源项目:

相关推荐
Frostpine1 天前
Docker 部署 Gitea 服务
后端
SimonKing1 天前
还在本地硬扛大模型?试试 Ollama Cloud,顺便把 OpenCode 也升级了
java·后端·程序员
CodeSheep1 天前
两位大佬相继离世,AI时代我们活得太着急了
前端·后端·程序员
Yu_Lijing1 天前
基于C++的《Head First设计模式》笔记——解释器模式
c++·设计模式·解释器模式
人间打气筒(Ada)1 天前
go实战案例:如何在 Go-kit 和 Service Meh 中进行服务注册与发现?
开发语言·后端·golang·istio·go-kit
lay_liu1 天前
springboot和springframework版本依赖关系
java·spring boot·后端
C++chaofan1 天前
RPC框架容错机制深度解析
java·开发语言·后端·性能优化·高并发·juc·容错机制
查古穆1 天前
FastAPI-passlib密码加密
后端·fastapi