SpringBoot整合Redis之使用@Cacheable注解

@Cacheable实现自动缓存,属性为value、key和condition:

参数 作用
value 缓存的名称
key 缓存的 key, SpEL 表达式
condition 缓存的条件

本文环境为SpringBoot2.X,以下为使用过程及个人理解:

添加依赖

<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-data-redis</artifactId>
	<exclusions>
		<exclusion>
			<groupId>io.lettuce</groupId>
			<artifactId>lettuce-core</artifactId>
		</exclusion>
	</exclusions>
</dependency>
<dependency>
	<groupId>redis.clients</groupId>
	<artifactId>jedis</artifactId>
</dependency>

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 {
    	/**
    	 * 在进行缓存数据与数据库数据同步时每次更新1000条,防止SQL语句拼接超长更新失败以及数据过多更新较慢问题
    	 */
    	public static final Integer DEFAULT_SYNC_LENGTH = 1000;
    	/**
    	 * 默认数据库中存储数据长度为200,后期根据数据库内存使用情况调整(调整步频为200)
    	 */
    	public static final Integer DEFAULT_MAX_LENGTH = 2000;
    	/**
    	 * 数据自增步长为1
    	 */
    	public static final Integer INCREMENT = 1;
    	/**
    	 * 数据自减步长为1
    	 */
    	public static final Integer REDUCTION = -1;
    	/**
    	 * 存储用户信息key
    	 */
    	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配置
     * 
     * @author single-聪
     * @date 2019年7月15日
     * @version 0.0.1
     */
    @Configuration
    @EnableCaching
    public class RedisConfig extends CachingConfigurerSupport {
    
    	/**
    	 * 使用Jackson2JsonRedisSerialize 替换默认序列化
    	 * 
    	 * @param redisConnectionFactory
    	 * @return
    	 */
    	@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);
    	}
    
    	/**
    	 * 配置自动化缓存使用的序列化方式以及过期时间
    	 * 
    	 * @return
    	 * @author single-聪
    	 * @date 2019年10月29日
    	 * @version 1.0.1
    	 */
    	@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();
    		// 配置数据存储序列化规则以及缓存过期时间(目前设置1天,后续根据实际情况修改)
    		configuration = configuration
    				.serializeValuesWith(
    						RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer))
    				.entryTtl(Duration.ofDays(1));
    		return configuration;
    	}
    
    	/**
    	 * 缓存过期时间自定义配置
    	 * 
    	 * @param redisConnectionFactory
    	 * @return
    	 * @author single-聪
    	 * @date 2019年10月29日
    	 * @version 1.0.1
    	 */
    	@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.CATEGORY, cacheConfiguration.entryTtl(Duration.ZERO)
    				.disableCachingNullValues().prefixKeysWith(RedisConstant.CATEGORY));
    
    		// 用户信息基本不会产生变化,且用户信息占用量不大,所以缓存时间较长
    		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序列化方式,但是你可以根据自己的需要设置每一类的序列化方式。

开始使用

我一般使用@Cacheable注解的时候会在service和mapper层之间再加一层操作,当然你可以直接使用在service层的方法上,这个随意。

@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的值

@Override
@Cacheable(value = RedisConstant.CATEGORY, key = "#root.methodName", unless = "#result == null")
public Object hot() {
	return kingdomMapper.selectWithScore();
}

上面这一种存储到缓存里面key的形式为:category:hot,其中hot的来源是方法名,这个涉及到key的相关取值,本文不介绍了。百度一大堆,但是其他的本人没用过。。。
下面来说一下@Cacheable注解的好处:

  • redisTemplate.opsForValue().get(key)方法不用写了,直接避免存数据和取数据时数据类型不一致情况的出现
  • 缓存为空判断不需要做了,省下大堆代码
  • 缓存有效期设置可以在配置文件中实现,具体人员只管使用,防止某类数据缓存时间变化时需要改动多处代码

下面说一下我在配置过程中遇到的坑,希望不要再犯:

  • 最初的时候我在RedisConfig配置文件中未配置cacheManager方法,同时RedisConstant常量类参数也没有加:,导致的情况是我在用Redis可视化工具查看数据的时候userid之间有两个:,虽然不影响使用,但是看着蛋疼啊!!
  • 当我配置了cacheManager方法之后,测试又发现userid之间没有:符号了,这不更是蛋疼吗?(如果某一天user展示数据突然要求加一个字段,难道我使用keys命令查询再删除?要知道生产环境明令禁止使用keys命令啊!)
  • 最终测试的结果是我前面的配置个人感觉更合适,使一类数据放到一组,方便管理。

如有高见敬请指出,在线更新。。

发布了43 篇原创文章 · 获赞 25 · 访问量 9万+

猜你喜欢

转载自blog.csdn.net/single_cong/article/details/102828227