Redis反序列化异常处理方法

问题背景

通过String类型存储,只不过Value是一个自定义对象。暂且叫这个类型为UserBo吧。
当我们通过redisTemplate.opsForValue().set(userId, userBo, 24, TimeUnit.HOURS)存入时并没有什么问题。
可是如果我们通过UserBo userBoCache = (UserBo)redisTemplate.opsForValue().get(userId);方式查看时日志中则会出现报错

Could not read JSON: Could not resolve type id 'com.dps.bo.user.UserBo' as a subtype of `java.lang.Object`

显然对于一个系统而言这是不稳定的。

问题分析

结合报错内容和我们所进行的redis操作可以判断,此问题的原因是我们存入的Value在取出时并不能被转换为UserBo类型的对象。我们暂且称这为反序列化问题。为什么会反序列化失败,原因大抵是我们选用了String类型来存储Object,所以在get时得到的是单纯的String类型,即便打印出来也是UserBo类型对象结构,但仍不能进行强制转换

解决方案

方案一修改spring-data-redis中RedisTemplate默认序列化方式
SpringBoot使用spring-data-redis,RedisTemplate默认的序列化方式是用org.springframework.data.redis.serializer.JdkSerializationRedisSerializer这个类来做序列化,而Redis有自己的序列化方式,所以冲突了 报出了上面的序列化异常的信息。

// 源码参考:spring-data-redis-2.7.2依赖中部分代码
private @Nullable RedisSerializer<?> defaultSerializer;

if (defaultSerializer == null) {
	defaultSerializer = new JdkSerializationRedisSerializer(
			classLoader != null ? classLoader : this.getClass().getClassLoader());
}

我们可以在使用到redisTemplate的类中加入下面的配置,指定使用的序列化方式,
将key序列化方式改为 StringRedisSerializer,将value序列化方式改为 GenericJackson2JsonRedisSerializer(Jackson2JsonRedisSerializer可以序列化成功,但是反序列化会失败,用GenericJackson2JsonRedisSerializer即可序列化也可反序列化) 。以下两种写法均可——一种是直接更改,一种是配置更改,没什么区别

    private RedisTemplate<String, QuestionTemplate> redisTemplate;
    //指定用redis的序列化方式进行序列化
    @Autowired(required = false)
    public void setRedisTemplate(RedisTemplate redisTemplate) {
        RedisSerializer stringSerializer = new StringRedisSerializer();//序列化为String
        //不能反序列化
        //Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);//序列化为Json
        GenericJackson2JsonRedisSerializer serializer = new GenericJackson2JsonRedisSerializer();
        redisTemplate.setKeySerializer(stringSerializer);
        redisTemplate.setValueSerializer(serializer);
        redisTemplate.setHashKeySerializer(stringSerializer);
        redisTemplate.setHashValueSerializer(serializer);
        this.redisTemplate = redisTemplate;
    }
@Configuration
public class RedisConfig {
    public RedisConfig() {
    }
    @Bean
    public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory connectionFactory) {
        RedisTemplate<Object, Object> redisTemplate = new RedisTemplate();
        redisTemplate.setConnectionFactory(connectionFactory);
        Jackson2JsonRedisSerializer serializer = new GenericJackson2JsonRedisSerializer(Object.class);
        ObjectMapper om = new ObjectMapper();
        om.setVisibility(PropertyAccessor.ALL, Visibility.ANY);
        om.enableDefaultTyping(DefaultTyping.NON_FINAL);
        serializer.setObjectMapper(om);
        redisTemplate.setKeySerializer(new StringRedisSerializer());
        redisTemplate.setValueSerializer(serializer);
        redisTemplate.setHashKeySerializer(new StringRedisSerializer());
        redisTemplate.setHashValueSerializer(serializer);
        return redisTemplate;
    }
}

PS:适应场景——有系统框架封装的Redis工具,且<String: Object>的String类型存取场景是使用的框架Redis工具 或者 系统框架中有RedisTemplate配置类(说白了就是统一配置到处使用。总不能你在哪个业务类里导入了个RedisTemplate你就得在这个业务类下面手动设置key和Value的序列化方式吧)

方案二自定义Object<->String的序列化与反序列化
UserBo在存入前通过Json工具序列化为String,取出后再通过序列化工具get为UserBo

redisTemplate.opsForValue().set(userId, JSON.toJSONString(userBo), 24, TimeUnit.HOURS);
UserBo userBoCache = JSON.parseObject(redisTemplate.opsForValue().get(userId), UserBo.class);

PS:适应场景——<String: Object>的String类型存取场景不多的情况(毕竟每个都这么写也挺烦的)

### 回答1: Fastjson是一个用于Java的高性能JSON处理库,可以方便地进行JSON的序列化和反序列化反序列化是将JSON格式的数据转换为Java对象的过程。在Redis中存储的value可以是任意类型的数据,包括字符串、数字、列表、哈希等。当我们从Redis中获取到value时,需要将其反序列化为对应的Java对象才能进行进一步的操作。 使用Fastjson进行反序列化Redis value的步骤如下: 1. 导入Fastjson库:首先需要将Fastjson的库添加到项目的依赖中,可以通过Maven或手动下载并导入相应的jar包。 2. 获取Redis value:从Redis中获取到的value可以通过Redis客户端的API来获取,一般是以字符串的形式返回。 3. 使用Fastjson进行反序列化:将获取到的Redis value以字符串的形式传递给Fastjson反序列化方法,然后指定要转换的Java对象的类型。Fastjson会自动将JSON字符串转换为对应类型的Java对象,可以直接进行操作。 示例代码如下: ```java import com.alibaba.fastjson.JSON; ... // 假设从Redis中获取到的value是一个包含学生信息的JSON字符串 String redisValue = redisClient.get("studentInfo"); // 使用FastjsonJSON字符串反序列化为Student对象 Student student = JSON.parseObject(redisValue, Student.class); // 可以直接对student对象进行操作 System.out.println(student.getName()); System.out.println(student.getAge()); ... ``` 通过上述步骤,我们可以使用Fastjson方便地将Redis中的value反序列化为Java对象,从而可以在应用程序中对其进行进一步的操作。需要注意的是,Fastjson在进行反序列化时,需要确保JSON字符串的格式与目标Java对象类型的属性一一对应,否则可能会出现异常。 ### 回答2: Fastjson是一款Java语言中的快速JSON解析库,能够高效地将JSON字符串转换为Java对象。对于Redis而言,它可以将Redis存储的JSON格式的value值反序列化为Java对象。 在实际应用中,我们通常会使用Fastjson提供的`parseObject()`方法进行反序列化操作。首先,我们需要将Redis中获取到的JSON字符串作为参数传递给该方法,Fastjson会根据JSON的结构和定义的目标对象类型将其转换成对应的Java对象。 具体的使用步骤如下: 1. 首先,我们需要从Redis中获取到存储的JSON字符串。 2. 使用Fastjson的`parseObject()`方法JSON字符串转换为Java对象。例如,如果我们希望将JSON字符串转换成User对象,可以使用如下代码: `User user = JSON.parseObject(jsonString, User.class);` 这里的`jsonString`是从Redis中获取到的JSON字符串,`User.class`表示目标对象的类型。 3. 对于嵌套的JSON结构,Fastjson也能够很好地支持反序列化操作。例如,假设User对象中包含了Address对象作为属性,我们同样可以使用`parseObject()`方法将其转换成对应的Java对象。 总结而言,Fastjson是一款高效的JSON解析库,可以帮助我们将Redis中存储的JSON格式的value值反序列化为Java对象。通过使用`parseObject()`方法,我们可以方便地将JSON字符串转换为目标对象,并且对于嵌套的JSON结构同样能够很好地支持。 ### 回答3: Fastjson是一种Java的JSON处理库,它提供了封装简单、速度快的特点,常用于序列化和反序列化JSON数据。在使用Fastjson进行反序列化时,可以将Redis中存储的JSON数据进行解析,转换为Java对象。 通常情况下,我们在往Redis中存储数据时会将数据转换为JSON格式,Fastjson可以将这些JSON数据转换为Java对象,方便我们在程序中进行操作。Fastjson提供了灵活的反序列化功能,可以将JSON字符串映射到Java对象的属性上,同时也支持从JSON数组中解析多个对象。 在反序列化Redis中的value时,我们可以首先使用Redis客户端从Redis中获取到相应的value,接着使用Fastjson的`parseObject`方法将这个value转换为Java对象。例如: ``` String redisValue = jedis.get("key"); // 从Redis中获取到value MyClass myObject = JSON.parseObject(redisValue, MyClass.class); // 使用Fastjson反序列化为Java对象 ``` 在上述代码中,`MyClass`是自定义的Java类,表示了一个与Redis中value对应的数据模型,`redisValue`则是获取到的Redis value对应的字符串。`JSON.parseObject`方法会根据字符串的内容将其转换为与`MyClass`结构相同的Java对象。 需要注意的是,在反序列化时,要确保将数据转换为正确的数据类型,否则可能会导致类型转换错误或其他异常。此外,Fastjson也提供了更多的反序列化选项和定制能力,可以根据具体需要进行配置。 总之,Fastjson可以作为一个强大的工具,用于反序列化Redis中的value数据,将之转换为Java对象,便于在程序中进行进一步的处理和操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值