杰克逊反序列化对象或数组
我有一个杰克逊问题。
有没有办法反序列化可能具有两种类型的属性,对于某些对象,它看起来像这样
"someObj" : { "obj1" : 5, etc....}
然后对于其他人来说,它显示为一个空数组,即
"someObj" : []
任何帮助是值得赞赏的!
谢谢!
我有一个杰克逊问题。
有没有办法反序列化可能具有两种类型的属性,对于某些对象,它看起来像这样
"someObj" : { "obj1" : 5, etc....}
然后对于其他人来说,它显示为一个空数组,即
"someObj" : []
任何帮助是值得赞赏的!
谢谢!
编辑:从Jackson 2.5.0开始,您可以使用DeserializationFeature.ACCEPT_EMPTY_ARRAY_AS_EMPTY_OBJECT来解决您的问题。
Bruce提供的解决方案存在一些问题/缺点:
以下是我对这个问题的“通用”解决方案:
public abstract class EmptyArrayAsNullDeserializer<T> extends JsonDeserializer<T> {
private final Class<T> clazz;
protected EmptyArrayAsNullDeserializer(Class<T> clazz) {
this.clazz = clazz;
}
@Override
public T deserialize(JsonParser jp, DeserializationContext ctxt) throws IOException {
ObjectCodec oc = jp.getCodec();
JsonNode node = oc.readTree(jp);
if (node.isArray() && !node.getElements().hasNext()) {
return null;
}
return oc.treeToValue(node, clazz);
}
}
那么你仍然需要为每个不同类型的自定义反序列化程序,但这更容易编写,并且你不会复制任何逻辑:
public class Thing2Deserializer extends EmptyArrayAsNullDeserializer<Thing2> {
public Thing2Deserializer() {
super(Thing2.class);
}
}
然后你像往常一样使用它:
@JsonDeserialize(using = Thing2Deserializer.class)
如果你找到一种方法来摆脱最后一步,即每种类型的实现1个自定义反序列化程序,我全都听;)
Jackson 当前没有内置配置来自动处理此特定情况,因此需要自定义反序列化处理。
下面是此类自定义反序列化可能的外观的示例。
import java.io.IOException;
import org.codehaus.jackson.JsonNode;
import org.codehaus.jackson.JsonParser;
import org.codehaus.jackson.JsonProcessingException;
import org.codehaus.jackson.Version;
import org.codehaus.jackson.annotate.JsonAutoDetect.Visibility;
import org.codehaus.jackson.annotate.JsonMethod;
import org.codehaus.jackson.map.DeserializationContext;
import org.codehaus.jackson.map.JsonDeserializer;
import org.codehaus.jackson.map.ObjectMapper;
import org.codehaus.jackson.map.module.SimpleModule;
public class JacksonFoo
{
public static void main(String[] args) throws Exception
{
// {"property1":{"property2":42}}
String json1 = "{\"property1\":{\"property2\":42}}";
// {"property1":[]}
String json2 = "{\"property1\":[]}";
SimpleModule module = new SimpleModule("", Version.unknownVersion());
module.addDeserializer(Thing2.class, new ArrayAsNullDeserializer());
ObjectMapper mapper = new ObjectMapper().setVisibility(JsonMethod.FIELD, Visibility.ANY).withModule(module);
Thing1 firstThing = mapper.readValue(json1, Thing1.class);
System.out.println(firstThing);
// output:
// Thing1: property1=Thing2: property2=42
Thing1 secondThing = mapper.readValue(json2, Thing1.class);
System.out.println(secondThing);
// output:
// Thing1: property1=null
}
}
class Thing1
{
Thing2 property1;
@Override
public String toString()
{
return String.format("Thing1: property1=%s", property1);
}
}
class Thing2
{
int property2;
@Override
public String toString()
{
return String.format("Thing2: property2=%d", property2);
}
}
class ArrayAsNullDeserializer extends JsonDeserializer<Thing2>
{
@Override
public Thing2 deserialize(JsonParser jp, DeserializationContext ctxt) throws IOException, JsonProcessingException
{
JsonNode node = jp.readValueAsTree();
if (node.isObject())
return new ObjectMapper().setVisibility(JsonMethod.FIELD, Visibility.ANY).readValue(node, Thing2.class);
return null;
}
}
(你可以利用DeserializationConfig.Feature.ACCEPT_SINGLE_VALUE_AS_ARRAY来强制输入始终绑定到集合,但考虑到当前如何描述问题,这可能不是我采用的方法。