Redis反序列化失败问题

发布于:2025-06-26 ⋅ 阅读:(15) ⋅ 点赞:(0)

redis反序列化类似于List ‘<Person’> 数据结构时,需要知道类型信息,
ObjectMapper.DefaultTyping.EVERYTHING 这句代码是关键点,让redis反序列化时知道List是什么类型

import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.JsonTypeInfo;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.json.JsonMapper;
import com.fasterxml.jackson.databind.jsontype.impl.LaissezFaireSubTypeValidator;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.cache.CacheManager;
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.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.Cursor;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.ScanOptions;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.RedisSerializationContext;
import org.springframework.data.redis.serializer.StringRedisSerializer;

import java.time.Duration;
import java.util.ArrayList;
import java.util.List;


/**
 * @ClassName RedisConfig
 * @Description redis配置类
 * @Date 2024/12/5 9:11
 */
@Slf4j
@EnableCaching  // 开启缓存
@Configuration
public class RedisConfig {
    // 定义一个Bean,名称为"redisTemplate",返回类型为RedisTemplate<String, Object>
    @Bean(name = "redisTemplate")
    public RedisTemplate<String, Object> getRedisTemplate(RedisConnectionFactory factory) {
        RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
        redisTemplate.setConnectionFactory(factory);
        StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
        Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(objectMapper(), Object.class);
        redisTemplate.setKeySerializer(stringRedisSerializer);
        redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
        redisTemplate.setHashKeySerializer(stringRedisSerializer);
        redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);
        redisTemplate.afterPropertiesSet();
        return redisTemplate;
    }

    @Bean("cacheManager")
    public CacheManager cacheManager(RedisConnectionFactory redisConnectionFactory) {
        Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer =
                new Jackson2JsonRedisSerializer<>(objectMapper(), Object.class);
        RedisCacheConfiguration cacheConfiguration = RedisCacheConfiguration.defaultCacheConfig()
                .entryTtl(Duration.ofMinutes(5))
                .disableCachingNullValues()
                .computePrefixWith(name -> name + ":")
                .serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(new StringRedisSerializer()))
                .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer));
        return RedisCacheManager.builder(redisConnectionFactory)
                .cacheDefaults(cacheConfiguration)
                .build();
    }

    private ObjectMapper objectMapper() {
        ObjectMapper objectMapper = JsonMapper.builder()
                .addModule(new JavaTimeModule())
                .build();
        objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
        objectMapper.activateDefaultTyping(LaissezFaireSubTypeValidator.instance,
                ObjectMapper.DefaultTyping.EVERYTHING, JsonTypeInfo.As.PROPERTY);
        return objectMapper;
    }

    @Bean
    @ConditionalOnProperty(name = "app.cache.clear-on-startup", havingValue = "true")
    public CommandLineRunner cacheClearRunner(RedisTemplate<String, Object> redisTemplate,
            @Value("${app.cache.clear-patterns:}") List<String> patterns) {

        return args -> {
            for (String pattern : patterns) {
                safeClearCache(redisTemplate, pattern);
            }
        };
    }

    private void safeClearCache(RedisTemplate<String, Object> redisTemplate, String pattern) {
        try {
            log.info("Clearing cache for pattern: " + pattern);
            int deleted = 0;
            Cursor<String> cursor = redisTemplate.scan(
                    ScanOptions.scanOptions().match(pattern).count(100).build());

            List<String> keysBatch = new ArrayList<>(100);
            while (cursor.hasNext()) {
                keysBatch.add(cursor.next());
                if (keysBatch.size() >= 100) {
                    redisTemplate.delete(keysBatch);
                    deleted += keysBatch.size();
                    keysBatch.clear();
                }
            }
            if (!keysBatch.isEmpty()) {
                redisTemplate.delete(keysBatch);
                deleted += keysBatch.size();
            }
            log.info("Cleared {} keys for pattern: {}", deleted, pattern);
        } catch (Exception e) {
            log.error("Cache clear error for pattern " + pattern + ": " + e.getMessage());
            log.error(e.getMessage(), e);
        }
    }
}

网站公告

今日签到

点亮在社区的每一天
去签到