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);

你可能感兴趣的:(Java,Java框架)