redis获取缓存对象bean时报:SerializationException: Could not read JSON: Could not resolve type
一、在单个应用内进行 存 取
工作中的问题场景:
将一个实体类对象set存入 redis中, 用的时候去get时,
报错提示:......SerializationException: Could not read JSON: Could not resolve type.......
明明实体类也实现 序列化接口了;
问题原因:
查看类中是否有除了get,set以外的其他方法,如果是注解开发,那就看看是不是有除了@Data以外的其他注解(如:@AllArgsConstructor 、@NoArgsConstructor);
在序列化时,只要类中方法有返回值,都会被序列化存入缓存中,存入时不会报错,
但是从缓存中取值时反序列化就会报错,因为除了对象属性字段有set方法,其他那些方法被序列号存入的值是没有set方法的
解决方案:
1、 设置redis反序列化时忽略未知属性
@Bean
public RedisTemplate<String, Object> redisTemplate(LettuceConnectionFactory lettuceConnectionFactory) {
Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(
Object.class);
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
// 此设置默认为true,就是在反序列化遇到未知属性时抛异常,这里设置为false,目的为忽略部分序列化对象存入缓存时误存的其他方法的返回值
objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
// 配置redisTemplate
RedisTemplate<String, Object> rtp = new RedisTemplate<String, Object>();
rtp.setConnectionFactory(lettuceConnectionFactory);
RedisSerializer<?> stringSerializer = new StringRedisSerializer();
// key序列化
rtp.setKeySerializer(stringSerializer);
// value序列化
rtp.setValueSerializer(jackson2JsonRedisSerializer);
// Hash key序列化
rtp.setHashKeySerializer(stringSerializer);
// Hash value序列化
rtp.setHashValueSerializer(jackson2JsonRedisSerializer);
rtp.afterPropertiesSet();
return rtp;
}
2、在实体类中只加 @Data注解,去掉其他会生成方法的注解或者去掉其他方法
二、多个应用交互,一个内进行 存,另一个进行 取
redis的默认通过io序列化和反序列化,
bug:当存取为vaule为实体类对象时,你发现抛标题里的异常了
原因:redis的默认通过io序列化和反序列化,于是给redis里存值时一切正常,但是另一个应用里去取值时,就会根据缓存里的类路径找寻对应的实体类进行反序列化,取得应用里当然不存在相同路径以及这个实体类,于是反序列化失败;
解决方案:
首先,这个bug属于 存 的一方的错误哦,
方法1:在redis全局配置那里配置redis的序列化和反序列化工具为fastjson这类工具,这个一搜有很多;
方法2:往redis里存的时候将实体类对象转json后再存,然后取时这样去取:
RecordBasicInfo basicInfo = JSON.parseObject(String.valueOf(info), RecordBasicInfo.class);
这里注意,转info为字符串时不要用转json工具,因为存的那一方已经转过了
更多推荐
所有评论(0)