首页 > 数据库 >使用WireShark分析使用RedisTemplate取不到值的问题

使用WireShark分析使用RedisTemplate取不到值的问题

时间:2022-10-24 15:02:23浏览次数:78  
标签:Redis CATALOG 存放 redisTemplate template 使用 new RedisTemplate WireShark

现象:
使用Java Redis客户端将数据存放到Redis后,使用redisTemplate却不出来
原因:
Java Redis客户端在将数据存放在Redis时,会对Key,Value,Field进行编码。从Redis中取数据时,如果Key、Field的编码和存放时不一样,就会取不出来。
就像中文乱码一样,同样的中文字符串,存放时编码格式不对,就不能被识别。

问题复盘及分析过程:

(1)使用下面代码将数据放到Redis中

stringRedisTemplate.opsForHash().put("CATALOG_DEV","KEY_INDEX_CATALOG",JsonUtil.list2json(list));

(2)使用方式1,可以hget到的value和存放时的一致:

stringRedisTemplate.opsForHash().get("CATALOG_DEV","KEY_INDEX_CATALOG")

(3)使用方式2,hget到的value是nulll:

redisTemplate.opsForHash().get("CATALOG_DEV","KEY_INDEX_CATALOG")

使用wireshark抓包:

方式1时,HGET命令发送到Redis服务器的指令:

使用WireShark分析使用RedisTemplate取不到值的问题_redis

方式2时,HGET命令发送到Redis服务器的指令:

使用WireShark分析使用RedisTemplate取不到值的问题_redis_02

 

可见,虽然代码中,Redis的Java客户端发送的key和field都是相同和字符串,但Redis接受到的Field却是不同的。
看看Redis服务器上存放的Field是什么样的:

使用WireShark分析使用RedisTemplate取不到值的问题_数据_03

 

 

看一个使用redisTemplate客户端存放的数据:

使用WireShark分析使用RedisTemplate取不到值的问题_redis_04

对比看下方式1存放到Redsi服务器上的数据

为什么有这个差异呢?来看看Spring容器中的stringRedisTemplate 和 redisTemplate实例有什么区别:
RedisTemplate的实例化代码如下:

@Bean
@SuppressWarnings({"rawtypes", "unchecked"})
public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory)
throws UnknownHostException {
RedisSerializer<String> stringSerializer = new StringRedisSerializer();
RedisTemplate<Object, Object> template = new RedisTemplate<>();
template.setConnectionFactory(redisConnectionFactory);
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
ObjectMapper om = new ObjectMapper();
om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(om);
// 设置value的序列化采用Jackson2JsonRedisSerializer
template.setValueSerializer(jackson2JsonRedisSerializer);
// 设置key的序列化采用StringRedisSerializer
template.setKeySerializer(stringSerializer);
// template.setHashKeySerializer(stringSerializer);
// template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.afterPropertiesSet();
return template;
}

HashKeySerializer、HashValueSerializer没有指定,应该会走默认,如下图所示:

使用WireShark分析使用RedisTemplate取不到值的问题_json_05

StringRedisTemplate的实例化代码如下:

@Bean
public StringRedisTemplate stringRedisTemplate(RedisConnectionFactory redisConnectionFactory) {
StringRedisTemplate template = new StringRedisTemplate();
template.setConnectionFactory(redisConnectionFactory);
return template;
}

可以看到,HashKeySerializer、HashValueSerializer都使用了StringRedisSerializer,与上面的情况一致。O了

使用WireShark分析使用RedisTemplate取不到值的问题_数据_06

 

扩展:
WireShark抓包利器:
​​​https://github.com/jzwinck/redis-wireshark​

链接:https://pan.baidu.com/s/1M9tXDN54FODU7NftkLfwKQ
提取码:kzhc
复制这段内容后打开百度网盘手机App,操作更方便哦

查看WireShark插件配置目录及已经配置插件的信息:

插件copy到目录:帮助--》关于--》文件夹--》Personal configuration

使用WireShark分析使用RedisTemplate取不到值的问题_redis_07

已经安装的插件列表:

使用WireShark分析使用RedisTemplate取不到值的问题_redis_08

 



标签:Redis,CATALOG,存放,redisTemplate,template,使用,new,RedisTemplate,WireShark
From: https://blog.51cto.com/u_15147537/5789791

相关文章