笔者在使用redis存放键值对时,发现存放的key和value多了\xac\xed\x00\x05t\x00&的前缀,笔者存放的是字符串,结果如下图所示,出现非预期的前缀。
![在这里插入图片描述](https://img-blog.csdnimg.cn/af6d3dd87abc4bf0964b70c3e8f6628c.png)
出现该问题的原因是, redis template
向redis存放使用java对象序列化的值,序列化方式和string的一般方式不同。
明明指定的另一个引文字符串作为key,但是实际存储后却多出来一串奇异字符串,这让笔者好奇心顿时升起来了,决定debug调试一下
![在这里插入图片描述](https://img-blog.csdnimg.cn/90c4cf4fc1c6458384b5696df6fb3b1e.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
从set()方法进入一路对execute()的重载方法向下跟踪,最终来到了一个T result = action.doInRedis(connToExpose);
代码处
![在这里插入图片描述](https://img-blog.csdnimg.cn/9244dd85947f46c9bab8de7c19188f3f.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
step into 进入doInRedis方法,
![在这里插入图片描述](https://img-blog.csdnimg.cn/eac6ec9e90154b12b781e8b7cb3c4558.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
可以看到在一开始传入的匿名对象就是这个ValueDeserializingRedisCallback
抽象类的一个匿名子类,并实现了set方法的最后一步,将数据存入redis。
来看看我的字符串key是怎么转化成byte数组的。对rawKey方法向下追踪…
-
获取Key的序列化器,然后对我们的key进行序列化
![在这里插入图片描述](https://img-blog.csdnimg.cn/d9dcbe39b61c45aaa2c40577bbe8ff23.png)
![在这里插入图片描述](https://img-blog.csdnimg.cn/43663a4f23da43cda14a1f31c8c8b67e.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_15,color_FFFFFF,t_70,g_se,x_16)
进入serialize(key)
方法,执行了JdkSerializationRedisSerializer
类中的serialize方法。继续向下跟踪…
![在这里插入图片描述](https://img-blog.csdnimg.cn/7239b4b26b8b4238b7f4296a8156f880.png)
![在这里插入图片描述](https://img-blog.csdnimg.cn/da5492d1ab1f4cbf82aee5823ad2607f.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_19,color_FFFFFF,t_70,g_se,x_16)
![在这里插入图片描述](https://img-blog.csdnimg.cn/d4c14a012568418db7ba20b6de5203db.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
最终!!!!,原来是使用了JDK自带的ObjectOutPutStream
将我们的String对象序列化成了byte[],
![在这里插入图片描述](https://img-blog.csdnimg.cn/83500015aa26459f82c976c9de80292b.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
来看看我们把byte数组转回字符串key发现在我原本的key之前确实拼接了乱码的字符串。
![在这里插入图片描述](https://img-blog.csdnimg.cn/b6d46bed66524e48a7120629894b79fd.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
RedisTemplate对Key的序列化了解了后,我们在看看对Value的序列化是怎么一回事?
![在这里插入图片描述](https://img-blog.csdnimg.cn/6c21e30fb6d5411387a99c1c4aba06d2.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_16,color_FFFFFF,t_70,g_se,x_16)
![在这里插入图片描述](https://img-blog.csdnimg.cn/bf9121ec3de04f3e95a86ab6fc8d2957.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_16,color_FFFFFF,t_70,g_se,x_16)
最终结果是同样的使用JDK自带的对象输出流对其进行序列化。
![在这里插入图片描述](https://img-blog.csdnimg.cn/881b83266725420d8d019186ee5f590a.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
在Key和Value都序列化后,将数据存入Redis。最终总结可以发现一切原因都是这个默认的JdkSerializationRedisSerializer
JDKRedis序列化器的序列化方式不能让人接受。
知道了问题所在,这个问题就已经解决了一半了。
我们能不能不使用这个默认的JDKRedis序列化器,自己实现一个或者换一个呢?
很幸运,Spring已经为我们提供了许多类型的序列化器了
![在这里插入图片描述](https://img-blog.csdnimg.cn/7ffd4fe809b148668be322cb4eef0bcd.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBAQWJzdHJhY3RlZA==,size_20,color_FFFFFF,t_70,g_se,x_16)
一般我们常用的序列化器是Jackson2JsonRedisSerializer
和StringRedisSerializer
这两种。看名字就知道Jackson2JsonRedisSerializer是将对象序列化成JSON形式的序列化器了。
-
Jackson2JsonRedisSerializer
:常用来直接序列化Value对象为JSON字符串。内部使用ObjectMapper
-
StringRedisSerializer
:常用来序列化Key,也可以用来序列化Value。
于是乎我们对RedisTemplate进行配置:
/**
* 如果key和value都使用的StringRedisSerializer序列化器,则推荐使用StringRedisTemplate
*
* 配置Redis的Key和Value的序列化器
* @param redisTemplate 从容器中获取RedisTemplate
* @return 修改后的RedisTemple
*/
@Bean
public RedisTemplate<Object, Object> redisStringTemplate(RedisTemplate<Object, Object> redisTemplate) {
StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
redisTemplate.setKeySerializer(stringRedisSerializer);
// 如果手动将Value转换成了JSON,就不要再用JSON序列化器了。
// redisTemplate.setValueSerializer(new Jackson2JsonRedisSerializer<>(Object.class));
redisTemplate.setValueSerializer(stringRedisSerializer);
return redisTemplate;
}
查验结果:key和value的序列化和反序列化都正常了。