我正在使用 Spring Cache,我传入一个键集合,返回一个实体列表。我想让缓存框架了解返回列表中的每个元素都将使用相应的代码进行缓存。目前,似乎关键是整个列表,如果我在随后的调用中缺少一个关键,它会尝试再次重新加载整个集合。
@Override
@Cacheable(value = "countries")
public List<Country> getAll(List<String>codes) {
return countryDao.findAllInCodes(codes);
}
另一种可能性是返回是一个映射,同样我希望缓存足够智能,只查询以前从未查询过的项目,也可以用它的键缓存每个项目。
@Override
@Cacheable(value = "countries")
public Map<String,Country> getAllByCode(List<String>codes) {
return countryDao.findAllInCodes(codes);
}
假设国家类如下所示:
class Country{
String code;
String fullName;
long id;
... // getters setters constructurs etc..
}
Spring Cache 可以做到这一点吗?
最佳答案
事实上,即使是 Spring's Caching Abstraction 也是可能的,但不是开箱即用 (OOTB)。本质上,您必须自定义 Spring 的 缓存基础设施(下文进一步解释)
默认,Spring 的 缓存基础设施使用整个 @Cacheable
方法参数作为缓存“键”,如 here 所述.当然,您也可以使用 SpEL 表达式 或自定义 KeyGenerator
实现来自定义 key 分辨率,如 here 所述。 .
不过,不会分解参数参数的集合或数组以及@Cacheable
方法的返回值 到单个缓存条目(即基于数组/集合或映射的键/值对)。
为此,您需要 Spring 的 CacheManager
(取决于您的缓存策略/提供者)和 Cache
接口(interface)的自定义实现。
NOTE: Ironically, this will be the 3rd time I have answered nearly the same question, first here, then here and now here, :-). Anyway...
我已经更新/清理了 my example (有点)这个帖子。
请注意 my example extends and customizes ConcurrentMapCacheManager
Spring 框架 本身提供。
理论上,您可以扩展/自定义任何 CacheManager
实现,例如 Spring Data Redis 中的 Redis,here ( source ),或 Spring Data GemFire 中的 Pivotal GemFire 的 CacheManager
,here (source)。 Pivotal GemFire 的开源版本是 Apache Geode,它有一个对应的 Spring Data Geode 项目,(Spring 中的 source for CacheManager Data Geode,与 SD GemFire 基本相同)。当然,您可以将此技术应用于其他缓存提供程序... Hazelcast、Ehcache 等。
然而,真正的工作是由 custom implementation 处理的。 (或更具体地说,base class ) Spring 的 Cache界面。
无论如何,希望来自 my example ,您将能够弄清楚您需要在应用程序中执行哪些操作才能满足应用程序的缓存要求。
此外,您可以应用相同的方法来处理 Maps
,但我将把它留给您作为练习,;-)。
希望这会有所帮助!
干杯, 约翰
关于java - 带有项目/实体集合的 Spring Cache,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44529029/