SpringCache的常用注解-@Cachable
- 可以标记在方法上,也可以标记在类上
- 参数key:缓存的key规则,可以使用SpringEL,默认是方法参数组合
- 参数value:缓存名称,可以指定多个
- 参数condition:满足condition才缓存
先看下面简单的例子:
点击查看代码
@Override
@Cacheable(value = {"query1"},key = "#root.methodName")
public List<TestUser> testCacheQuery() {
return testUserMapper.selectList(new QueryWrapper<>());
}
当然单纯使用方法名作为key不是一个好主意,因为这个方法的所有结果都使用了同一个key,为key加上参数会更精准。
- 乱码的解决
从上面的图可以看出,在redis中存储的是乱码,这就涉及到序列化的问题,其实在使用RedisTemplate也会遇到类似的问题。这时需要进行序列化配置,请参考如下代码(其中还包含了过期时间的配置):
点击查看代码
@Bean
public RedisCacheManager cacheManagerTTL(RedisConnectionFactory connectionFactory) {
RedisCacheConfiguration config = instanceConfig(60L);
return RedisCacheManager.builder(connectionFactory)
.cacheDefaults(config)
.transactionAware()
.build();
}
private RedisCacheConfiguration instanceConfig(Long ttl) {
Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class);
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
objectMapper.registerModule(new JavaTimeModule());
// 去掉各种@JsonSerialize注解的解析
objectMapper.configure(MapperFeature.USE_ANNOTATIONS, false);
// 只针对非空的值进行序列化
objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
// 将类型序列化到属性json字符串中
objectMapper.activateDefaultTyping(LaissezFaireSubTypeValidator.instance ,
ObjectMapper.DefaultTyping.NON_FINAL, JsonTypeInfo.As.PROPERTY);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
return RedisCacheConfiguration.defaultCacheConfig()
.entryTtl(Duration.ofSeconds(ttl))
//.disableCachingNullValues()
.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer));
}
同时需要在缓存的方法配cacheManager,与配置中方法名称保持一致。
点击查看代码
@Override
@Cacheable(value = {"query1"},key = "#root.args[0]",cacheManager = "cacheManagerTTL")
public TestUser testCacheQuery(Integer id) {
return testUserMapper.selectById(id);
}