编辑:应你们中的一些人的要求,我发布了我的真实数据结构和类。我未能让您更容易理解问题。
我在解决杰克逊null
JSON 数组元素时遇到了问题,最后出现了一个异常,我将在本文后面报告。
我有一个来自第 3 方提供商的冗长 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");
}
...
在极端综合中,我试图将问题图解如下:
TinyMap
类有一个模块成员,它是ArrayList
一个TinyMapModule
对象TinyMapModule
类有一个宏成员,它是ArrayList
一个TinyMapMacro
对象TinyMapMacro
类映射具有我严格需要的唯一属性的对象。
现在,当我尝试将上面的 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
我已经阅读了很多关于 Jackson 注释、自定义反序列化器等的信息,但我似乎想出的唯一解决方案是将TinyMap.modules
成员 from转换ArrayList<TinyMapModules>
为ArrayList<Object>
并使用自定义反序列化器来处理 NullType 对象。
通过这种选择(如果可行,因为我还没有测试过)我会失去TinyMap.modules
成员的语义价值,我不会被迫这样做。
还有其他方法可以解决空元素问题吗?
编辑:
重要的是我可以在这个结构中保留非空对象的序数位置,所以我绝对应该至少用可用作占位符的东西替换空元素。