springcache中@Cacheable注解中只用value属性,造成缓存不正确

场景还原

 

springcache集成redis做缓存的时候,
使用@cacheable(value="cacheName"),
标注一个getA(id)方法和getB(id)方法,那么在默认生成Key的策略SimpleKeyGenerator中,key的生成方式就是用传递的参数拼接生成的
springcache中@Cacheable注解中只用value属性,造成缓存不正确_第1张图片

那么id来作为缓存的key,但是id肯定有重复的情况;
我本来以为value属性就是用来区分不同业务对象的一个机制,
{
我本来的理解:
在getA(id)方法上添加@cacheable(value="AA"),
在getB(id)方法上添加@cacheable(value="BB"),
这样就可以将两个id一样的A方法和B方法给区分开了;
}
但是,调用getA(1111)后,调用getB(1111),从缓存里取到的居然是A执行方法的结果!
可见,我的自以为是不对的,于是就陷入苦苦求索的道路
 

造成原因

value属性,确实是指定了缓存的名称,但是并没有强行将value不同的缓存值加以区分(这个是有道理的,因为有些时候,需要把不同业务属性的实体存在一个缓存里,这种情况肯定是有的,所以如果强行通过value区分的话,上述需求反而实现不了了)
注:从Spring 4.1开始,value缓存注释的属性不再是强制性的,新定义的一个实行cachename是value的别名


解决方法

既然value无法作为区分缓存的条件,我们可以从key上下手;
1.无脑的修改就是在key上多拼凑一些标志行的字符串,比如 key = "#root.methodName"+#id
2.相对稳妥的继续就是修改SimpleKeyGenerator的生成策略
 

@EnableCaching
@Configuration
public class RedisCacheConfig extends CachingConfigurerSupport {

    @Autowired
    private JedisConnectionFactory jedisConnectionFactory;

    @Bean
    public RedisTemplate redisTemplate() {
        StringRedisTemplate redisTemplate = new StringRedisTemplate(jedisConnectionFactory);

        Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class);
        ObjectMapper om = new ObjectMapper();
        om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
        om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        jackson2JsonRedisSerializer.setObjectMapper(om);

        redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
        redisTemplate.afterPropertiesSet();
        return redisTemplate;
    }

    @Bean
    public CacheManager cacheManager() {
        String[] cacheNames = {"app_default", "users", "blogs", "goods", "configs", "info"};
        RedisCacheManager redisCacheManager = new RedisCacheManager(redisTemplate(), Arrays.asList(cacheNames));
        redisCacheManager.setDefaultExpiration(86400);
        return redisCacheManager;
    }

    @Bean
    public Cache cache() {
        return cacheManager().getCache("app_default");
    }

    @Bean
    @Override
    public KeyGenerator keyGenerator() {
        return (target, method, objects) -> {
            StringBuilder sb = new StringBuilder();
            sb.append(target.getClass().getName());
            sb.append("::" + method.getName() + ":");
            for (Object obj : objects) {
                sb.append(obj.toString());
            }
            return sb.toString();
        };
    }
}


 

 

你可能感兴趣的:(java,springCache,redis,Cacheable,缓存失败)