我尝试使用两个缓存来存储具有相同ID的两个不同项目User和Group。我希望我应该得到不同的值,因为它们是两个不同的缓存。但事实并非如此。
@Bean
public JedisConnectionFactory jedisConnectionFactory() {
LOG.debug("redis host: {}, redis port: {}", redisHost, redisPort);
JedisConnectionFactory factory = new JedisConnectionFactory();
factory.setUseSsl(true);
factory.setHostName(redisHost);
factory.setPort(redisPort);
factory.setUsePool(true);
factory.setPassword(primaryKey);
return factory;
}
/**
*
* @return
*/
@Bean(name = "redisTemplate")
public RedisTemplate redisTemplate() {
RedisTemplate template = new RedisTemplate();
template.setConnectionFactory(jedisConnectionFactory());
return template;
}
/**
*
* @return
*/
@Bean
public CacheManager cacheManager() {
RedisCacheManager manager = new RedisCacheManager(redisTemplate());
manager.setCacheNames(Arrays.asList("users", "groups"));
return manager;
}
@Cacheable(cacheNames = "users", key = "#id")
public User findUserById(String id) {
return null;
}
@Cacheable(cacheNames = "users", key = "#user.id")
public User updateUser(User user) {
return user;
}
@Cacheable(cacheNames = "groups", key = "#id")
public Group findGroupById(String id) {
return null;
}
@CachePut(cacheNames = "groups", key = "#group.id")
public Group updateGroup(Group group) {
return group;
}
JUnit的。
service.removeCaches();
User user = new User();
user.setName("oldUser");
user.setId("1");
// USER
assertNull(service.findUserById("1"));
service.updateUser(user);
User userCached = service.findUserById("1");
LOG.debug(userCached);
assertNotNull(userCached);
assertEquals(userCached.getName(), "oldUser");
// GROUP
assertNull(service.findGroupById("1")); // <== here, im actually getting user instead of null. cache has been cleared before the test starts.
也许我在这里误解了不同缓存的概念。或者即使在不同的缓存中,密钥也应该是唯一的?
日志
Results :
Tests in error:
testCache(TestRedisServiceImpl): User cannot be cast to Group
这基本上表示从缓存中返回用户。
答案 0 :(得分:3)
实际上,CacheManager.setCacheNames()
是为同一个缓存提供别名,它不是两个不同的缓存。
您可以查看API here:
void setCacheNames(Collection cacheNames)指定的集合 此CacheManager的静态名称&#39;静态&#39;模式。
您可以认为 CacheManager就像一个Map
对象,其中一个键只存储一个对象(作为值),因此当您尝试使用相同的键插入第二个对象时,第一个对象将被替换为第二个对象。
所以,正在发生的事情是user
中的CacheManager
对象被group
对象替换,因为它们都存储有相同的密钥。
要在缓存管理器中使缓存键唯一,您可以使用cacheManager.setUsePrefix()
,这样每个键都将以相应的cachename为前缀,并且不会替换值。