2

我想要动态缓存名称,而 spring 4.1 允许

从 Spring 4.1 开始,缓存注解的 value 属性不再是强制性的,因为无论注解的内容如何,​​CacheResolver 都可以提供此特定信息。

请注意我是如何偏执地设置cacheResolver所有可能的级别的:

@Cacheable(cacheResolver = "defaultCacheResolver")
@CacheConfig(cacheResolver = "defaultCacheResolver")
public interface GatewayRepository extends CrudRepository<Gateway, Integer> {
    @Cacheable(cacheResolver = "defaultCacheResolver")
    Gateway findByBulkId(int bulkId);
}

Spring 4.1.5 仍然无法通过错误验证配置:Caused by: java.lang.IllegalStateException: No cache names could be detected on 'public abstract skunkworks.data.Gateway skunkworks.repos.GatewayRepository.findByBulkId(int)'. Make sure to set the value parameter on the annotation or declare a @CacheConfig at the class-level with the default cache name(s) to use. at org.springframework.cache.annotation.SpringCacheAnnotationParser.validateCacheOperation(SpringCacheAnnotationParser.java:240)

4

1 回答 1

3

我认为您应该在代码中的某处指定缓存名称。

在基本用法中,缓存名称在@Cacheable、@CachePut 或@CacheEvict 注释中给出。

@Cacheable(cacheNames = "myCache")

您还可以在 @CacheConfig 中指定它,这是一个类级别的注释。

@CacheConfig(cacheNames = "myCache")

如果需要更灵活的缓存机制,可以使用CacheResolver。在这种情况下,您应该创建自己的 CacheResolver。这些方面的东西:

public class CustomCacheResolver implements CacheResolver {

    private final CacheManager cacheManager;

    public CustomCacheResolver(CacheManager cacheManager){
        this.cacheManager = cacheManager;
    }

    @Override
    public Collection<? extends Cache> resolveCaches(CacheOperationInvocationContext<?> context) {
        Collection<Cache> caches = new ArrayList<>();
        if(context.getTarget().getClass() == GatewayRepository.class){
            if(context.getMethod().getName().equals("findByBulkId")){
                caches.add(cacheManager.getCache("gatewayCache"));
            }
        }

        return caches;
    }
}

在这一步中,缓存名称为

网关缓存

它仅在 cacheresolver 中定义,可以在注释端省略。

在这一步之后,您应该注册 CacheResolver:

@Configuration
@EnableCaching
public class CacheConfiguration extends CachingConfigurerSupport {

    @Bean
    @Override
    public CacheManager cacheManager() {
         // Desired CacheManager
    }

    @Bean
    @Override
    public CacheResolver cacheResolver() {
        return new CustomCacheResolver(cacheManager());
    }
}

作为最后一步,您应该在 @Cacheable、@CachePut、@CacheConfig 等注释之一中指定 CustomCacheResolver。

@Cacheable(cacheResolver="cacheResolver")

您可以在此处查找代码示例

于 2017-04-13T17:01:25.830 回答