关于Redis的问题探讨(二):Range方法返回的对象是LinkeHashMap以及转换办法

上一篇:关于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)

运行测试:

 可以发现转换成功了

Tags: