Spring Cache注解无效:可能是类内方法调用注解缓存的方法
in 未分类 with 0 comment

Spring Cache注解无效:可能是类内方法调用注解缓存的方法

in 未分类 with 0 comment

产生原因 最近做一简单的缓存,使用到spring cache框架,集成EhCache。其间多方面努力解决了多cacheManger冲突后,在实际使用中发现使用缓存注解无效,讲道理有点小崩溃,毕竟测试时并没有问题。 但是仔细一想,测试的时候并没有在类内方法去调用注解的缓存的方法,可能问题就出现在这里。 形如:` @Cacheable public String getUserNme(Long userId){

return userService.getUserName(userId); } //这时候通过此方法调用缓存,导致注解无效 public String getUserNameChache(Long userId){ return etUserNme(userId); }

`

解决方案:

1.不使用注解的方式,直接取 Ehcache 的 CacheManger 对象,把需要缓存的数据放到里面,类似于使用 Map,缓存的逻辑自己控制;或者可以使用redis的缓存方式去添加缓存;

2.把方法A和方法B放到两个不同的类里面,例如:如果两个方法都在同一个service接口里,把方法B放到另一个service里面,这样在A方法里调B方法,就可以使用B方法的缓存。

3.通过((UserService)AopContext.currentProxy().xxx()的方法获取当前类的代理类;

4.通过ApplicationContext获取当前类的代理对象

为什么缓存没有被正常创建?? 因为@Cacheable 是使用AOP 代理实现的 ,通过创建内部类来代理缓存方法,这样就会导致一个问题,类内部的方法调用类内部的缓存方法不会走代理,不会走代理,就不能正常创建缓存,所以每次都需要去调用数据库。

@Cacheable 的一些注意点 1、因为@Cacheable 由AOP 实现,所以,如果该方法被其它注解切入,当缓存命中的时候,则其它注解不能正常切入并执行,@Before 也不行,当缓存没有命中的时候,其它注解可以正常工作

2、@Cacheable 方法不能进行内部调用,否则缓存无法创建  

@Cacheable标注的方法,如果其所在的类实现了某一个接口,那么该方法也必须出现在接口里面,否则cache无效。 具体的原因是, Spring把实现类装载成为Bean的时候,会用代理包装一下,所以从Spring Bean的角度看,只有接口里面的方法是可见的,其它的都隐藏了,自然课看不到实现类里面的非接口方法,@Cacheable不起作用。

解决办法:把待cache的方法移到接口里面。

另外衍生两个小问题:

  1. @Cacheable放接口里面可以吗?答案是:不行。
  2. 如果某一个Bean并没有实现任何接口,@Cacheable标注的方法有什么要求?    答案是public即可。这种Bean也被Spring产生了代理, 看得到的只有public方法。

本质是Spring代理的问题,很多的基础设施可能都会遇到类似的问题。比如安全,事务,日志等等。

0评论