Jackson:如何反序列化JSON数组中的null元素

时间:2012-10-31 15:46:58

标签: java json jackson deserialization pojo

编辑:根据你们中的一些人的要求,我发布了我的真实数据结构和类。我没有尝试让你更容易理解这个问题。

我在使用Jackson null JSON数组元素进行解析时遇到问题,最后会有一个例外,我将在本文稍后报告。

我有来自第三方提供商的冗长的JSON字符串:

[
  null,
  null,
  [
    {
      "t" : {
        "id" : 0,
        "it" : "Attuazione"
      },
      "c" : "DM.10.0.2",
      "l" : "Macro 2",
      "v0" : 0,
      "scid" : "1",
      "m" : "10",
      "sam" : 0,
      "v1" : 100,
      "smcm" : true,
      "w" : "OD.10.4",
      "st" : {
        "auto_formula" : {
          "ia" : "($x > 0 ? 1 : 0)",
          "id" : "($x > 0 ? 1 : 0)",
          "bit" : "($x > 0 ? 1 : 0)",
          "oa" : "($x > 0 ? 1 : 0)",
          "od" : "($x > 0 ? 1 : 0)",
          "dm" : "($x > 0 ? 1 : 0)"
        },
        "id" : 0,
        "css_rule" : "containerTypeOnOff",
        "default_measure_unit" : null,
        "it" : "On \/ Off",
        "widget" : {
          "id" : 0,
          "caption" : {
            "it" : "Etichetta di testo"
          }
        }
      }
    }
  ],
  null,
  null,
  null,
  null,
  null
]

我的Java类集合如下。

这是最外面的容器:

public class TinyMap {
    ArrayList<TinyMapModule> modules;

    public TinyMap() { ...  }

    public ArrayList<TinyMapModule> getModules() { ... }

    /**
     * @param modules the modules to set
     */
    @SuppressWarnings("unchecked")
    public void setModules(ArrayList<TinyMapModule> modules) { ... }

    public void appendNewMacro(TinyMapModule m) { ... }

}

这是第一级容器:

public class TinyMapModule {
    private ArrayList<TinyMapMacro> macros;

    /**
     * C'tor 
     */
    public TinyMapModule() { ... }

    /**
     * @return ArrayList<TinyMapMacro> 
     */
    public ArrayList<TinyMapMacro> getMacros() { ... }

    /**
     * @param macros the macros to set
     */
    @SuppressWarnings("unchecked")
    public void setMacros(ArrayList<TinyMapMacro> macros) { ... }

}

这是最里面的水平:

public class TinyMapMacro {
    private String c; 
    private String l;
    private String m;
    private MacroType t;
    private String w;

    private static Logger logger;

    static {
        logger = Logger.getLogger(TinyMapMacro.class);
        DOMConfigurator.configure("log4j.xml");
    }

    /**
     * C'tor 
     */
    public TinyMapMacro() { ... }

    public static class MacroType {
        private int id;
        private String it; // Italian

        public int getId() { ... }
        public void setId(int id) { ... }
        public String getIt() { ... }
        public void setIt(String it) { ... }
    }

    public String getC() { ... }

    public void setC(String c) { ... }

    public String getL() { ... }

    public void setL(String l) { ... }

    public String getM() { ... }

    public void setM(String m) { ... }

    public MacroType getT() { ... }

    public void setT(MacroType t) { ... }

    public String getW() { ... }

    public void setW(String w) { ... }

    public int getTypeId() { ... }

    public String getLocalizedTypeString() { ... }

}

最后,这是我的反序列化代码:

ObjectMapper mapper = new ObjectMapper();
mapper.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);
mapper.disable(DeserializationFeature.FAIL_ON_NULL_FOR_PRIMITIVES);
ObjectReader reader = mapper.reader(TinyMapModule.class);
try {
    logger.info("Instantiating ObjectMapper");
    TinyMap tm = new TinyMap();
    tm.buildTinyMap(reader.readValues(jsonString.replaceAll("null", "[{\"name\":\"\"}]")));
    for (int i = 0; i < tm.getModules().size(); i++) {
        if (tm.getModules().get(i) == null) 
            continue;
        for (int j = 0; j < tm.getModules().get(i).getMacros().size(); j++) {
            if (tm.getModules().get(i).getMacros().get(j) == null) 
                continue;
            System.out.println(String.format("c: %s", tm.getModules().get(i).getMacros().get(i).getC()));
            System.out.println(String.format("l: %s", tm.getModules().get(i).getMacros().get(i).getL()));
            System.out.println(String.format("m: %s", tm.getModules().get(i).getMacros().get(i).getM()));
            System.out.println(String.format("t.id: %d", tm.getModules().get(i).getMacros().get(i).getTypeId()));
            System.out.println(String.format("t.it: %s", tm.getModules().get(i).getMacros().get(i).getLocalizedTypeString()));
            System.out.println(String.format("w: %s", tm.getModules().get(i).getMacros().get(i).getW()));
        }
    }
    logger.info("Execution terminated");
} 
... 

在极端综合中,我试图按如下方式对问题进行模式化:

  1. TinyMap类有一个模块成员,ArrayListTinyMapModule个对象
  2. TinyMapModule类有一个宏成员,ArrayListTinyMapMacro个对象
  3. TinyMapMacro类使用我严格需要的唯一属性映射对象。
  4. 现在,当我尝试将上面的JSON字符串反序列化为我的对象时,我得到一个异常,告诉我JSON无法反序列化:

    Exception in thread "main" com.fasterxml.jackson.databind.RuntimeJsonMappingException: Can not deserialize instance of it.asystelsrl.hcsbridge.communication.TinyMapModule out of START_ARRAY token
     at [Source: java.io.StringReader@3c50507; line: 1, column: 2]
        at com.fasterxml.jackson.databind.MappingIterator.next(MappingIterator.java:122)
        at it.asystelsrl.hcsbridge.communication.TinyMap.buildTinyMap(TinyMap.java:47)
        at it.asystelsrl.hcsbridge.tests.TestTinyMap.main(TestTinyMap.java:69)
    Caused by: com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of it.asystelsrl.hcsbridge.communication.TinyMapModule out of START_ARRAY token
     at [Source: java.io.StringReader@3c50507; line: 1, column: 2]
        at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:164)
        at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.java:599)
        at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.java:593)
        at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromArray(BeanDeserializer.java:531)
        at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:141)
        at com.fasterxml.jackson.databind.MappingIterator.nextValue(MappingIterator.java:188)
        at com.fasterxml.jackson.databind.MappingIterator.next(MappingIterator.java:120)
        ... 2 more
    

    我已经阅读了很多关于杰克逊注释,自定义反序列化器等的内容,但我似乎提出的唯一解决方案是将TinyMap.modules成员从ArrayList<TinyMapModules>转换为ArrayList<Object>和使用自定义反序列化器来处理NullType对象。

    通过这种选择(如果可行的话,因为我还没有测试过它)我将失去TinyMap.modules成员的语义价值,我不会被迫。

    还有其他方法可以解决null元素问题吗?

    编辑:

    重要的是我可以保留此结构中非空对象的序号位置,所以我绝对应该至少用可用作占位符的东西替换null元素。

2 个答案:

答案 0 :(得分:2)

这里的问题是我使用杰克逊制造了很多混乱。

我开始理解它正在阅读this article by ProgrammerBruce

我试图将一个Collection反序列化为一个不是的东西,正如StaxMan在他的评论中所建议的那样。

感谢ProgrammerBruce提供的有用而且清晰的文章。

答案 1 :(得分:1)

您仍然只在示例JSON中显示theBeans属性,而不是完整的MyContainer对象。而且您的错误听起来好像是在尝试将给定的JSON解析为MyContainer

为了确定,您可以发布完整的JSON对象和反序列化代码吗?

杰克逊理解无效的POJO就好了。