Spring缓存/ spring存储库,驱逐多个密钥

Sau*_*aul 4 spring spring-data spring-cache

我有两种方法来获取具有两个不同参数的实体.我还有一个使用其中一个参数的保存方法.如何在两个提取键下从缓存中逐出实体?例如见下文:

@Cacheable  
public User getByUsername(String username);

@Cacheable  
public User getByEmail(String email);


@CacheEvict(key="#entity.username")
User save(User entity); 
Run Code Online (Sandbox Code Playgroud)

在上面,对getByEmail的调用将返回过时日期.

Joh*_*lum 11

当然,有几种选择,但像往常一样,Spring有你的背.

最简单,最简单的方法是在方法上利用Spring的@Caching注释save,就像这样......

@Caching(evict = {
  @CacheEvict(cacheNames = "Users", key="#user.name"),
  @CacheEvict(cacheNames = "Users", key="#user.email")
})
User save(User user);
Run Code Online (Sandbox Code Playgroud)

为了您的参考,我创建了一个示例测试类,演示了这里的工作原理.

你会注意到我在我的UserRepository上使用Spring的 Cache Abstraction注释模仿了你上面的例子.在这种情况下,我的仓库由Pivotal GemFire支持,但任何数据存储都可以使用.我使用Spring作为我的缓存提供程序,使用Spring的ConcurrentMapCacheManager,但当然,任何缓存提供程序都可以.ConcurrentMap

我的测试用例继续保存新的 User,确保用户存储尚未缓存.然后,测试继续执行查询方法(findByName,findByEmail),确保在每种情况下适当地缓存用户实体.然后,我从基础数据存储中删除实体,并确保实体仍然被缓存.最后,在真实的时刻,我修改实体,重新保存实体,断言实体的存储是因为所有条目都已从缓存中"逐出".

作为另一种优化,您还可以尝试在这种情况下将@CachePut注释与2个@CacheEvict注释相结合,这可以根据新的更新实体恢复缓存,例如...

@Caching(
  evict = {
    @CacheEvict(cacheNames = "Users", key="#a0.name", beforeInvocation = true),
    @CacheEvict(cacheNames = "Users", key="#a0.email", beforeInvocation = true)
  },
  put = {
    @CachePut(cacheNames = "Users", key="#result.name"),
    @CachePut(cacheNames = "Users", key="#result.email")
  }
)
User save(User user);
Run Code Online (Sandbox Code Playgroud)

注意:注意注释上beforeInvocation属性的用户@CacheEvict以及@CachePuts

但是,您可能希望根据需要将实体延迟添加到缓存中.

虽然,您可以假设实体经常被访问/使用,因为save您的repo上的方法刚刚被调用,因此依赖您的底层数据存储(例如GemFire)来设置额外的驱逐(基于溢出)/到期(基于LRU)设置,从而更好地管理您的系统资源(例如内存),同时仍保持最佳的应用程序性能.

值得深思.

希望这可以帮助.