springboot 缓存反序列化失败:
项目使用 redis 来做缓存,使用 springboot 的 @Cacheable 缓存数据时,可以成功存入 redis, 但是在第二次请求命中缓存时,就会出现一下错误:
java.lang.ClassCastException: java.util.LinkedHashMap cannot be cast to ...
经定位,确定了错误原因:
原因:springboot 的缓存使用 jackson 来做数据的序列化与反序列化,如果默认使用 Object 作为序列化与反序列化的类型,则其只能识别 java 基本类型,遇到复杂类型时,jackson 就会先序列化成 LinkedHashMap ,然后再尝试强转为所需类别,这样大部分情况下会强转失败。此时就需要指定序列化方式为:
GenericJackson2JsonRedisSerializer,
指定后,在序列化时,会将类名存入到序列化后的 json 字符串中,如:
{"@class": "com.example.SpecialClass", "id" : 1, .... }
这样在取出缓存时,springboot 就可以自动根据 @class 对应的字段找到对应的类进行反序列化了
配置:
@Bean
public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {
RedisTemplate<String, Object> redisTemplate = new RedisTemplate<String, Object>();
redisTemplate.setConnectionFactory(factory);
Jackson2JsonRedisSerializer<Object> serializer = new Jackson2JsonRedisSerializer<>(Object.class);
redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer());
redisTemplate.setDefaultSerializer(serializer);
return redisTemplate;
}
参考: spring-redistemplate-serialise-multiple-model-classes-into-json-need-to-use-mu
本文讨论了SpringBoot使用Redis缓存时遇到的反序列化失败问题。当使用@Cacheable存储复杂类型数据时,由于默认的Jackson序列化机制导致ClassCastException。解决方法是明确指定序列化策略,使Jackson在序列化时包含类信息,从而在反序列化时能正确转换回所需类。
9722





