@Cacheable实现自动缓存,属性为value、key和condition:
参数 | 作用 |
---|---|
value | 缓存的名称 |
key | 缓存的 key, SpEL 表达式 |
condition | 缓存的条件 |
本文环境为SpringBoot2.X,以下为使用过程及个人理解:
org.springframework.boot
spring-boot-starter-data-redis
io.lettuce
lettuce-core
redis.clients
jedis
SpringBoot2.X使用lettuce,但是个人习惯加上其他原因所以换成了jedis。
redis:
port: 6379
jedis:
pool:
max-active: 500
max-wait: 50000
max-idle: 500
min-idle: 0
timeout: 50000
host: www.baidu.com
cache:
type: redis #缓存类型
redis:
cache-null-values: false #不缓存null数据
time-to-live: 50000ms #超时时间
use-key-prefix: false #不使用前缀
host换成你自己的IP即可
RedisConstant.java
public class RedisConstant {
/**
* 数据自增步长为1
*/
public static final Integer INCREMENT = 1;
/**
* 数据自减步长为1
*/
public static final Integer REDUCTION = -1;
/**
* 存储用户信息前缀
*/
public static final String USER = "user:";
}
鉴于篇幅,此处只放USER一个常量,这个常量类的作用:全局使用,一改全改,注意,如果你不想麻烦就一定要将USER常量的:
留着,对于后续操作缓存提供了太多的方便!!
RedisConfig.java
配置类,大用!即使不使用@Cacheable这个类也该有
package org.config;
import java.sql.SQLException;
import java.time.Duration;
import java.util.HashMap;
import java.util.Map;
import javax.sql.DataSource;
import org.constant.RedisConstant;
import org.springframework.cache.CacheManager;
import org.springframework.cache.annotation.CachingConfigurerSupport;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheConfiguration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.cache.RedisCacheWriter;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.RedisSerializationContext;
import org.springframework.data.redis.serializer.StringRedisSerializer;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.PlatformTransactionManager;
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
/**
* 缓存Redis配置
*/
@Configuration
@EnableCaching
public class RedisConfig extends CachingConfigurerSupport {
/**
* 使用Jackson2JsonRedisSerialize 替换默认序列化
*/
@Bean
public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {
RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();
// 使用Jackson2JsonRedisSerialize 替换默认序列化
Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<Object>(
Object.class);
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
// 设置value的序列化规则和 key的序列化规则
redisTemplate.setKeySerializer(new StringRedisSerializer());
redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
// hash参数序列化方式
redisTemplate.setHashKeySerializer(new StringRedisSerializer());
redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);
// 缓存支持回滚(事务管理)
redisTemplate.setEnableTransactionSupport(true);
redisTemplate.setConnectionFactory(redisConnectionFactory);
redisTemplate.afterPropertiesSet();
return redisTemplate;
}
// 配置事务管理器
@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) throws SQLException {
return new DataSourceTransactionManager(dataSource);
}
/**
* 配置自动化缓存使用的序列化方式以及过期时间
*/
@Bean
public RedisCacheConfiguration redisCacheConfiguration() {
Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<Object>(
Object.class);
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
RedisCacheConfiguration configuration = RedisCacheConfiguration.defaultCacheConfig();
configuration = configuration
.serializeValuesWith(
RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer))
.entryTtl(Duration.ofDays(1));
return configuration;
}
/**
* 缓存过期时间自定义配置
*/
@Bean
public CacheManager cacheManager(RedisConnectionFactory redisConnectionFactory) {
// 设置CacheManager的值序列化方式为Jackson2JsonRedisSerializer,默认就是使用StringRedisSerializer序列化key,JdkSerializationRedisSerializer序列化value
Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<Object>(
Object.class);
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
// 配置value序列化方式为Jackson2JsonRedisSerializer,key序列化方式采用默认的StringRedisSerializer
RedisCacheConfiguration cacheConfiguration = RedisCacheConfiguration.defaultCacheConfig().serializeValuesWith(
RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer));
// 每一类信息进行缓存配置
Map<String, RedisCacheConfiguration> redisCacheConfigurationMap = new HashMap<>();
// 用户信息
redisCacheConfigurationMap.put(RedisConstant.USER, cacheConfiguration.entryTtl(Duration.ofDays(10))
.disableCachingNullValues().prefixKeysWith(RedisConstant.USER));
// 初始化一个RedisCacheWriter
RedisCacheWriter redisCacheWriter = RedisCacheWriter.nonLockingRedisCacheWriter(redisConnectionFactory);
RedisCacheConfiguration defaultCacheConfig = RedisCacheConfiguration.defaultCacheConfig().serializeValuesWith(
RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer));
// 设置默认超过期时间是1天(短时间的已经做了其他的处理,不会采用注解形式加入缓存)
defaultCacheConfig.entryTtl(Duration.ofDays(1));
// 初始化RedisCacheManager
RedisCacheManager cacheManager = new RedisCacheManager(redisCacheWriter, defaultCacheConfig,
redisCacheConfigurationMap);
return cacheManager;
}
}
嗯,redisCacheConfiguration是我看的别人的配置抄的,设置默认的过期时间为1天
重点是cacheManager方法:在本文中我写了两个方法,但是在实际开发中你可以根据RedisConstant里面常量的数量及需求自行增加,这样就不需要在方法上面修改了(万一一个key在多处调用岂不是蛋疼?),而且我是为了省事所以所有的都使用的Jackson2JsonRedisSerializer序列化方式,但是你可以根据自己的需要设置每一类的序列化方式。
上述配置中.disableCachingNullValues()
配置要根据个人系统决定!此配置会导致@Cacheable
如果返回null
会报错,缓存拒绝存储null
值:
java.lang.IllegalArgumentException: Cache 'user:' does not allow 'null' values. Avoid storing null via '@Cacheable(unless="#result == null")' or configure RedisCache to allow 'null' via RedisCacheConfiguration.
at org.springframework.data.redis.cache.RedisCache.put(RedisCache.java:139)
不建议进行上述配置,一方面报错很难受,另一方面数据如果从数据库中查不到大概率数据库中根本就不存在此条记录,所以让此后的数据继续去查询数据库没什么意义,还不如在缓存中放置一个null
值的数据,指定过期时间即可。
但是有一些特殊的数据需要进行上述设置,这个根据个人系统决定。
我一般使用@Cacheable注解的时候会在service和mapper层之间再加一层操作,当然你可以直接使用在service层或者dao层的接口上,这个随意。
@Override
@Cacheable(value = RedisConstant.USER, key = "#id", condition = "#bool==true", unless = "#result == null")
public UserShow getUser(String id, boolean bool) {
return userMapper.userShow(id);
}
上面这一种存储到缓存里面key的形式为:user:userId
,其中userId为方法参数id的值,绝大多数情况下建议使用keyGenerator
,编译过程中可能会解析不到参数名,导致缓存时key
拼接错误。
@Override
@Cacheable(value = RedisConstant.CATEGORY, key = "#root.methodName", unless = "#result == null")
public Object hot() {
return kingdomMapper.selectWithScore();
}
上面这一种存储到缓存里面key的形式为:category:hot
,其中hot的来源是方法名,这个涉及到key的相关取值,本文不介绍了。百度一大堆,但是其他的本人没用过。。。
下面来说一下@Cacheable注解的好处:
下面说一下我在配置过程中遇到的坑,希望不要再犯:
:
,导致的情况是我在用Redis可视化工具查看数据的时候user
和id
之间有两个:
,虽然不影响使用,但是看着蛋疼啊!!user
和id
之间没有:
符号了,这不更是蛋疼吗?(如果某一天user展示数据突然要求加一个字段,难道我使用keys命令查询再删除?要知道生产环境明令禁止使用keys命令啊!)如有高见敬请指出,在线更新。。