我打算使用Spring @Cacheable注释来缓存调用方法的结果。
但是这个实现在某种程度上对我来说并不是非常“安全”。据我所知,返回的值将由底层缓存引擎缓存,并在调用Spring evict方法时被删除。
我需要一个在加载新值之前不会破坏旧值的实现。这是必需的,以下场景应该有效:
这怎么可能?
答案 0 :(得分:2)
如果@Cacheable方法抛出异常,您可以通过对Google Guava的最小扩展轻松实现您提供旧值的要求。
使用以下示例配置
@Configuration
@EnableWebMvc
@EnableCaching
@ComponentScan("com.yonosoft.poc.cache")
public class ApplicationConfig extends CachingConfigurerSupport{
@Bean
@Override
public CacheManager cacheManager() {
SimpleCacheManager simpleCacheManager = new SimpleCacheManager();
GuavaCache todoCache = new GuavaCache("todo", CacheBuilder.newBuilder()
.refreshAfterWrite(10, TimeUnit.MINUTES)
.maximumSize(10)
.build(new CacheLoader<Object, Object>() {
@Override
public Object load(Object key) throws Exception {
CacheKey cacheKey = (CacheKey)key;
return cacheKey.method.invoke(cacheKey.target, cacheKey.params);
}
}));
simpleCacheManager.setCaches(Arrays.asList(todoCache));
return simpleCacheManager;
}
@Bean
@Override
public KeyGenerator keyGenerator() {
return new KeyGenerator() {
@Override
public Object generate(Object target, Method method, Object... params) {
return new CacheKey(target, method, params);
}
};
}
private class CacheKey extends SimpleKey {
private static final long serialVersionUID = -1013132832917334168L;
private Object target;
private Method method;
private Object[] params;
private CacheKey(Object target, Method method, Object... params) {
super(params);
this.target = target;
this.method = method;
this.params = params;
}
}
}
CacheKey用于公开SimpleKey属性的单一目的。 Guavas refreshAfterWrite将配置刷新时间而不会使缓存条目到期。如果使用@Cacheable注释的方法抛出异常,则缓存将继续提供旧值,直到由于maximumSize被驱逐或被成功方法响应中的新值替换。您可以将refreshAfterWrite与expireAfterAccess和expireAfterAccess结合使用。
答案 1 :(得分:0)
我在阅读Spring代码时可能错了,尤其是org.springframework.cache.interceptor.CacheAspectSupport#execute(org.springframework.cache.interceptor.CacheOperationInvoker, org.springframework.cache.interceptor.CacheAspectSupport.CacheOperationContexts)
,但我相信抽象并不能提供你所要求的内容。
CacheErrorHandler
机制仅处理与缓存调用相关的异常。总而言之,在我看来,你要求的是特定于用例的,因此不是抽象应该/应该提供的东西。