Redis Range方法输出LinkedHashMap,如何有效转换?
- 内容介绍
- 文章标签
- 相关推荐
本文共计640个文字,预计阅读时间需要3分钟。
上一篇文章:关于Redis的问题探讨(一):为什么存放集合偏向于转String后存放而非直接存+在上篇文章中,发现了一个问题,明明存的是一个对象的集合,为什么通过range方法返回的却是LinkedHashMap?
上一篇:关于Redis的问题探讨(一):为何存放集合偏向于转String后存放而非直接存
在上篇中发现了一个问题,明明存的是一个对象的集合,为什么通过range方法返回的却是LinkeHashMap
于是通过debug看看源码,了解执行流程,找一下原因
这是笔者通过debug画出的大致流程:
注意看8中,mapObject方法,这里就是返回LinkeHashMap的原因,这里重点解析一下mapObject方法:
protected Object mapObject(JsonParser p, DeserializationContext ctxt) throws IOException {
// 获取字段名称,也就是key
String key1 = p.getText();
// 执行前该token是FIELD_NAME,也就是字段名称
// 然后执行nextToken()后,变为了VALUE_STRING,也就是String类型的字段值
// 但是如果值不为String类型,那就会在后面又执行一次nextToken,变为VALUE_NUMBER_INT,也就是整数型字段值
p.nextToken();
Object value1 = this.deserialize(p, ctxt);
String key2 = p.nextFieldName();
if (key2 == null) {
LinkedHashMap<String, Object> result = new LinkedHashMap(2);
result.put(key1, value1);
return result;
} else {
// 解释在上面
p.nextToken();
// 获取value值
Object value2 = this.deserialize(p, ctxt);
String key = p.nextFieldName();
LinkedHashMap result;
if (key == null) {
result = new LinkedHashMap(4);
result.put(key1, value1);
result.put(key2, value2);
return result;
} else {
result = new LinkedHashMap();
result.put(key1, value1);
result.put(key2, value2);
do {
p.nextToken();
result.put(key, this.deserialize(p, ctxt));
} while((key = p.nextFieldName()) != null);
return result;
}
}
}
首先就是这个p也就是JsonParser对象,他是通过JsonFactory的createParser方法创建的,其作用是解析Json字符串,因此在执行之前mapObject之前就已经将value解析并存入JsonParser中了。
JsonParser解释参考:blog.csdn.net/LBWNB_Java/article/details/120003565
然后执行流程可以参考注释
以上就是为何返回LinkeHashMap的原因,那么如何解决呢
这里就要用到ObjectMapper,在range方法的流程图中也看到其实也使用了他
通过ObjectMapper的convertValue方法将LinkeHashMap转为目标对象:
@Test public void testRange() { String key = "right_push_all_01"; List<LinkedHashMap<String, Object>> linkedHashMapList = redisService.lRange(key, 0, -1); ObjectMapper objectMapper = new ObjectMapper(); List<ThisIsDTO> thisIsDTOList = objectMapper.convertValue(linkedHashMapList, new TypeReference<List<ThisIsDTO>>() { }); for (ThisIsDTO thisIsDTO : thisIsDTOList) { System.out.println(thisIsDTO.getAge()); } }
注意:因为是集合存放,所以一定要new TypeReference<List<ThisIsDTO>>来定义转换的类型
如果只是单个对象,可以直接 objectMapper.convertValue(linkedHashMap, ThisIsDTO.class)
运行测试:
可以发现转换成功了
本文共计640个文字,预计阅读时间需要3分钟。
上一篇文章:关于Redis的问题探讨(一):为什么存放集合偏向于转String后存放而非直接存+在上篇文章中,发现了一个问题,明明存的是一个对象的集合,为什么通过range方法返回的却是LinkedHashMap?
上一篇:关于Redis的问题探讨(一):为何存放集合偏向于转String后存放而非直接存
在上篇中发现了一个问题,明明存的是一个对象的集合,为什么通过range方法返回的却是LinkeHashMap
于是通过debug看看源码,了解执行流程,找一下原因
这是笔者通过debug画出的大致流程:
注意看8中,mapObject方法,这里就是返回LinkeHashMap的原因,这里重点解析一下mapObject方法:
protected Object mapObject(JsonParser p, DeserializationContext ctxt) throws IOException {
// 获取字段名称,也就是key
String key1 = p.getText();
// 执行前该token是FIELD_NAME,也就是字段名称
// 然后执行nextToken()后,变为了VALUE_STRING,也就是String类型的字段值
// 但是如果值不为String类型,那就会在后面又执行一次nextToken,变为VALUE_NUMBER_INT,也就是整数型字段值
p.nextToken();
Object value1 = this.deserialize(p, ctxt);
String key2 = p.nextFieldName();
if (key2 == null) {
LinkedHashMap<String, Object> result = new LinkedHashMap(2);
result.put(key1, value1);
return result;
} else {
// 解释在上面
p.nextToken();
// 获取value值
Object value2 = this.deserialize(p, ctxt);
String key = p.nextFieldName();
LinkedHashMap result;
if (key == null) {
result = new LinkedHashMap(4);
result.put(key1, value1);
result.put(key2, value2);
return result;
} else {
result = new LinkedHashMap();
result.put(key1, value1);
result.put(key2, value2);
do {
p.nextToken();
result.put(key, this.deserialize(p, ctxt));
} while((key = p.nextFieldName()) != null);
return result;
}
}
}
首先就是这个p也就是JsonParser对象,他是通过JsonFactory的createParser方法创建的,其作用是解析Json字符串,因此在执行之前mapObject之前就已经将value解析并存入JsonParser中了。
JsonParser解释参考:blog.csdn.net/LBWNB_Java/article/details/120003565
然后执行流程可以参考注释
以上就是为何返回LinkeHashMap的原因,那么如何解决呢
这里就要用到ObjectMapper,在range方法的流程图中也看到其实也使用了他
通过ObjectMapper的convertValue方法将LinkeHashMap转为目标对象:
@Test public void testRange() { String key = "right_push_all_01"; List<LinkedHashMap<String, Object>> linkedHashMapList = redisService.lRange(key, 0, -1); ObjectMapper objectMapper = new ObjectMapper(); List<ThisIsDTO> thisIsDTOList = objectMapper.convertValue(linkedHashMapList, new TypeReference<List<ThisIsDTO>>() { }); for (ThisIsDTO thisIsDTO : thisIsDTOList) { System.out.println(thisIsDTO.getAge()); } }
注意:因为是集合存放,所以一定要new TypeReference<List<ThisIsDTO>>来定义转换的类型
如果只是单个对象,可以直接 objectMapper.convertValue(linkedHashMap, ThisIsDTO.class)
运行测试:
可以发现转换成功了

