在缓存的使用场景中经常需要使用到过期事件,某些情况我们需要对缓存的过期事件进行监听并进行自己的操作,本文即为SpringBoot2.0整合Redis过期事件监听配置。


  • 修改缓存参数
    修改缓存的conf文件,设置参数notify-keyspace-events “Ex”,默认是无参数的,将参数设置为Ex即可。
  • 缓存配置
    在配置文件中进行缓存连接参数配置此处略过。
    缓存序列化配置如下:
package org.pet.king.config;
import java.sql.SQLException;
import java.time.Duration;
import java.util.HashMap;
import java.util.Map;
import javax.sql.DataSource;
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.lettuce.LettuceConnectionFactory;
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 {

	/**
	 * redisTemplate 序列化使用的jdkSerializeable, 存储二进制字节码, 所以自定义序列化类
	 * 
	 * @param lettuceConnectionFactory
	 * @return
	 */
	@Bean
	public RedisTemplate<Object, Object> redisTemplate(LettuceConnectionFactory lettuceConnectionFactory) {
		// // 解决每次传入的都是一个lettuceConnectionFactory实际上使用同一个分区问题
		// RedisStandaloneConfiguration redisStandaloneConfiguration =
		// lettuceConnectionFactory
		// .getStandaloneConfiguration();
		// LettuceClientConfiguration lettuceClientConfiguration =
		// lettuceConnectionFactory.getClientConfiguration();
		// // 默认分区为0
		// redisStandaloneConfiguration.setDatabase(0);
		// LettuceConnectionFactory connectionFactory = new
		// LettuceConnectionFactory(redisStandaloneConfiguration,
		// lettuceClientConfiguration);
		// connectionFactory.afterPropertiesSet();

		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(lettuceConnectionFactory);
		redisTemplate.afterPropertiesSet();
		return redisTemplate;
	}

	// 配置事务管理器
	@Bean
	public PlatformTransactionManager transactionManager(DataSource dataSource) throws SQLException {
		return new DataSourceTransactionManager(dataSource);
	}

我一般使用的序列化方式都为Jackson2JsonRedisSerializer,当然你可以根据自己的需要设置自己的序列化方式。缓存的序列化配置在这里就结束了,但是需要注意一点(很多人应该使用不到):
代码中注释的那部分代码,本意是我想让不同的redisTemplate使用不同的分区(默认是db0),但是当我配置了几个redisTemplate之后发现所有的redisTemplate使用的分区是最终的那个分区(按照加载顺序最终加载的是哪个分区所有的redisTemplate就使用哪个分区),配置这个的目的是:缓存中基本所有数据都会过期,但是我只想要监听我需要做数据处理的那一部分过期key,其他的监听了没意义还要做逻辑判断浪费资源,但是我发现一个服务中没法随意切换db分区(将服务分开或许是一种解决方式,将自己想要监听的过期key全部放入db(n)中),注意一个服务中虽然不能随意切换,但是可以切换!!但是切换之后所有的redisTemplate操作的分区就会变成新切换的分区,易造成前面加入缓存中的数据查询不到的情况(慎用!!!!)

  • 实现MessageListener接口
package org.pet.king.config;

import org.pet.king.entity.MessageCollect;
import org.pet.king.entity.MessageFans;
import org.pet.king.entity.Publish;
import org.pet.king.entity.UserShow;
import org.pet.king.service.PetService;
import org.pet.king.service.PublishService;
import org.pet.king.service.UserService;
import org.springframework.data.redis.connection.Message;
import org.springframework.data.redis.connection.MessageListener;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import lombok.extern.slf4j.Slf4j;

/**
 * Redis缓存key过期监听
 * 
 * @author single-聪
 * @date 2019年11月22日
 * @version 0.0.1
 */
@Slf4j
@Component
public class RedisExpireListener implements MessageListener {

	private RedisTemplate<Object, Object> redisTemplate;

	private UserService userService;

	@Override
	public void onMessage(Message message, byte[] pattern) {
		log.info("监听key过期事件[{}]...[{}]", message, pattern);
		String expireKey = message.toString();
		if (expireKey.startsWith("ip")) {
			// IP数据过期事件过多,第一个就将其排除,不进行后续操作
			return;
		} else {
			log.info("根据key过期前缀调用相应接口");
		}
	}

	public void setRedisTemplate(RedisTemplate<Object, Object> redisTemplate) {
		this.redisTemplate = redisTemplate;
	}

	public void setUserService(UserService userService) {
		this.userService = userService;
	}
}

这里需要注意一点,在这个实现类里面不能进行@Autowired或者@Resource注入,在这个类会注入失败,导致你调用这个接口的方法的时候会报空指针异常(实际上是userService未注入,它是null,而不是参数为null),所以我这里使用的是私有属性,通过set方法设置,设置的地方在下面这个类中。

  • 配置缓存过期监听
package org.pet.king.config;

import org.pet.king.service.PetService;
import org.pet.king.service.PublishService;
import org.pet.king.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.listener.ChannelTopic;
import org.springframework.data.redis.listener.RedisMessageListenerContainer;

@Configuration
public class RedisMessageListener {
	@Autowired
	private RedisConnectionFactory redisConnectionFactory;
	
	@Autowired
	private RedisTemplate<Object, Object> redisTemplate;
	
	@Autowired
	private UserService userService;

	@Bean
	public RedisMessageListenerContainer redisMessageListenerContainer() {
		RedisMessageListenerContainer container = new RedisMessageListenerContainer();
		container.setConnectionFactory(redisConnectionFactory);
		// 解决bean注入失败问题
		RedisExpireListener redisExpireListener = new RedisExpireListener();
		redisExpireListener.setRedisTemplate(redisTemplate);
		redisExpireListener.setUserService(userService);
		// 监听缓存key过期
		container.addMessageListener(redisExpireListener, new ChannelTopic("__keyevent@0__:expired"));
		return container;
	}
}

上述配置即可监听缓存db0区的数据,至于具体的表达式可以自行上网百度。