gpt4 book ai didi

json - JacksonPolymorphicDeserialization : JsonMappingException

转载 作者:行者123 更新时间:2023-12-04 04:30:14 29 4
gpt4 key购买 nike

我以为我有一个父类参数,它有2个子类 ComboParameter IntegerParameter

@JsonSubTypes({
@JsonSubTypes.Type(value = IntegerParameter.class, name = "integerParam"),
@JsonSubTypes.Type(value = ComboParameter.class, name = "comboParam")
})
@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = As.WRAPPER_OBJECT)
public abstract class Parameter {
String regEx;
}


@JsonTypeName("integerParam")
public class IntegerParameter extends Parameter {
}

@JsonTypeName("comboParam")
public class ComboParameter extends Parameter {
List<String> values;
}

我有一个具有属性参数的类
class A {
@JsonUnwrapped
Parameter parameter;
}

对象 A的序列化引发异常

com.fasterxml.jackson.databind.JsonMappingException: Unwrapped property requires use of type information: can not serialize without disabling SerializationFeature.FAIL_ON_UNWRAPPED_TYPE_IDENTIFIERS



而且,如果我删除注释 @JsonUnwrapped,我将拥有一个类似于
{
parameter:{
integerParam:{
regEx: regExVal
}
}
}

我需要的是一个像这样的json:
{
integerParam:{
regEx: regExVal
}
}

NB 我正在使用Jackson 2.4.4

最佳答案

不要认为有解决此问题的简便方法。但是以下是一些解决方法(两种情况都为Gist demo):

选项1:在属性上方添加@JsonIgnore,并在顶级bean中添加@JsonAnyGetter。易于实现,但在bean中具有静态ObjectMapper不太好,并且必须将此代码复制到每个带有Parameter属性的代码中

public class A {

@JsonIgnore
Parameter parameter;

// can put ObjectMapper and most of this code in util class
// and just use JacksonUtils.toMap(parameter) as return of JsonAnyGetter

private static ObjectMapper mapper = new ObjectMapper();

/************************ Serialization ************************/

@JsonAnyGetter
private Map<String, Object> parameterAsMap(){
return mapper.convertValue(parameter, Map.class);
}

/************************ Deserialization **********************/

@JsonAnySetter
private void parameterFromMap(String key, JsonNode value) {
try {
parameter = mapper.readValue(String.format("{\"%s\":%s}", key,value),
Parameter.class);
} catch (IOException e) {
e.printStackTrace();
}
}
}

选项2: @JsonIgnore属性,并为根 A类注册 自定义序列化器/反序列化器
SimpleModule module = new SimpleModule();
module.addSerializer(A.class, new ASerializer());
module.addDeserializer(A.class, new ADeserializer());
mapper.registerModule(module);

不能使用A类之上的 @JsonSerialize,因为内部ObjectMapper的序列化器和反序列化器也将使用此批注,但是您需要将其设置为使用默认的序列化器/解串器,而不是递归地使用它。或者,如果您确实想要注释,则可以实现类似 https://stackoverflow.com/a/18405958/1032167的方法

序列化器+反序列化器看起来像这样( 未优化的,只是概念的证明):
    /************************ Serialization ************************/

public static class ASerializer extends JsonSerializer<A> {
private static ObjectMapper m = new ObjectMapper();

@Override
public void serialize(A value, JsonGenerator gen,
SerializerProvider serializers) throws IOException {
Map defaults = m.convertValue(value, Map.class);
Map params = m.convertValue(value.getParameter(), Map.class);
defaults.putAll(params);
gen.writeObject(defaults);
}

}

/************************ Deserialization **********************/

public static class ADeserializer extends JsonDeserializer<A> {
private static ObjectMapper m = new ObjectMapper();
private static String[] subtipes = {"integerParam", "comboParam"};

public ADeserializer() {
m.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);
}

@Override
public A deserialize(JsonParser p, DeserializationContext ctxt)
throws IOException {

TreeNode node = m.readTree(p);

A a = m.convertValue(node, A.class);

// hardcoded , probably can be done dynamically
// with annotations inspection
for (String key : subtipes) {
TreeNode value = node.get(key);
if (value != null) {
String json = String.format("{\"%s\":%s}", key, value);
a.setParameter(m.readValue(json, Parameter.class));
break;
}
}

return a;
}
}

通用解串器将很难编写。但是无论如何根据问题正文这个问题都是关于序列化的。

关于json - JacksonPolymorphicDeserialization : JsonMappingException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28993035/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com