Spring AOP内部方法调用代理失效

原生CGLib内部方法互相调用时可以代理,但基于CGLib的Spring AOP却代理失效

背景:

校准程序校准一分钟前的数据,假设18:01:00开始校准 17:59:00 - 18:00:00,此时因订单批次行更新modifyTime由18:00:00修改为18:00:01,
mysql更新即时可见,es有延迟,查询17:59:00 - 18:00:00范围数据es有、mysql无,从es插入mysql数据,主表插入为saveorupdate操作,
其他表为save操作,主键冲突应回滚,但由于Spring aop内部方法调用无法被代理(事务)导致数据未回滚,旧状态覆盖了新状态,数据不一致。

1.原因分析

原生CGLib代理类,相当于重写原生类方法,且只保留代理类的对象proxyBean,所有调用都走proxyBean,所以可以被代理。

Spring AOP无法拦截内部方法调用,Spring会保留原生类的对象bean以及代理类的对象proxyBean,
proxyBean进行了切面增强处理,调用proxyBean相当于:

before
invoke(bean,method)
after

这样处理会导致内部方法调用时代理失效,传入的是原生类的对象bean,所以内部方法调用不可以被代理。

2.解决方案

  1. 修改类,不要出现内部“自调用”的情况,这是Spring文档中推荐的“最佳”方案;
  2. 若一定要使用“自调用”,那么this.doSomething()替换为代理类对象:((CustomerService) AopContext.currentProxy()).doSomething();
  3. 此时需要修改spring的aop配置:@EnableAspectJAutoProxy(exposeProxy = true);
  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值