当前位置: 首页 > news >正文

@Transactional注解失效

@Transactional注解失效

在工作中,因为对该注解的原理了解不深导致出现bug,现进行记录。
场景复现:
在同一个类中,一个方法调用了另一个被@Transactional注解修饰的方法,此时事务是不生效的。我们进行try-catch将异常进行捕捉,信息为:java.lang.IllegalStateException: Transaction synchronization is not active。它标识事务未起效。
接下类我们来看看为什么它不生效

1.SpringAop的原理

用平时最常用的AspectJ来举例,它是使用Cglib进行反射生成代理对象进行我们的切面处理。抽象的来说,这就相当于将我们的切面操作的代码直接移植到原有的类上面去,再生成一个新的类。达到我们想要的切面效果。

2.@Transactional注解

当我们进入Transactional发现它就是一个注解,没有任何实现方法,那就说明它肯定是被框架所检测,然后在框架中去具体实现它的功能。

3.举例
public class TransactionAop {public void doWithoutTransaction() {log.info("doWithoutTransaction this: {}", this.getClass().getName());doWithTransaction();}@Transactionalpublic void doWithTransaction() {try{TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() {@Overridepublic void afterCommit() {log.info("doWithTransaction this: {}", this.getClass().getName());}});}catch (Exception e) {log.info(e.getMessage());}}
}

通过VisualVm只管的看到同一个类TransactionAop生成了两个对象

在这里插入图片描述

第一点:第一个对象都是通过Cglib生成的代理对象。第二个对象是Spring初始化时通过反射创建的原始对象。在初始化对象时,会去扫描整个类,标记那些方法被增强了。
第二代:Spring在调用一个对象的方法时,会去检查拦截器链。即使一个方法没有被增强,也会走一遍这个流程,去判断拦截器链为空,不会发生AOP事件。这里Spring做了缓存,方法只有在第一次被调用时会去检查,后续直接走缓存。

在调用doWithoutTransaction()时,通过match方法,检查这个方法是否被匹配上了。
那怎么匹配的,匹配的是什么东西呢?
在类AbstractFallbackTransactionAttributeSource中存在一个ConcurrentHashMap缓存,Spring会所有扫描到的被@Transiactional修饰的方法缓存起来。——Map<Object, TransactionAttribute> attributeCache

这里的Key是下面这个对象,所以具体的关系就是方法和类绑定起来成为一个Key

public final class MethodClassKey implements Comparable<MethodClassKey> {private final Method method;@Nullableprivate final Class<?> targetClass;}

可以看到在调用doWithoutTransaction时

在这里插入图片描述

在这里插入图片描述

没有匹配上,后续在找拦截器时就没有

而调用doWithTransaction时就匹配上了,可以看到他的事务等级为开启事务
在这里插入图片描述

4.那我就是需要让同一个类中的调用生效怎么办?
 // 获取当前代理对象,并调用事务方法
TransactionAop selfProxy = (TransactionAop) AopContext.currentProxy();
selfProxy.doWithTransaction();

相关文章:

  • 用c语言实现——一个交互式的中序线索二叉树系统,支持用户动态构建、线索化、遍历和查询功能
  • 超详细Kokoro-82M本地部署教程
  • 自定义类型-结构体(二)
  • 本地大模型工具深度评测:LM Studio vs Ollama,开发者选型指南
  • Java多线程(超详细版!!)
  • C++STL——priority_queue
  • 【Redis】基础命令数据结构
  • 【C++】string类
  • Linux进程间通信(四)之补充【日志】
  • 算法训练营第十三天|226.翻转二叉树、101. 对称二叉树、 104.二叉树的最大深度、111.二叉树的最小深度
  • 使用 librosa 测量《忘尘谷》节拍速度
  • 人形机器人量产元年开启,AI与物理世界深度融合
  • 局域网常用的测速工具,Iperf3使用教程
  • 数仓-如何保障指标的一致性
  • U盘制作系统盘(含U盘恢复)
  • dockerfile编写入门
  • 正式部署abp vnext应用程序时,如何生成openiddict.pfx证书文件
  • Proser:在使用中改进
  • 21、DeepSeekMath论文笔记(GRPO)
  • 如何更改typora图片存储位置
  • 特朗普将启的中东行会如何影响伊美核谈判?专家分析
  • 航行警告:渤海海峡黄海北部执行军事任务,禁止驶入
  • 巴防空系统击落印度无人机,印称巴方违反停火协议
  • 贵州省总工会党组成员、副主席梁伟接受审查调查
  • 高龄老人骨折后,生死可能就在家属一念之间
  • 道指跌逾100点,特斯拉涨近5%