如何使用Jackson的objectMapper反序列化接口字段?

Nam*_*man 23 java json jackson

ObjectMapperreadValue(InputStream in, Class<T> valueType)功能需要班级.但是如果我在内部传递的类有一些接口作为数据成员,我该如何使用它.

虽然我可以理解这个例外背后的原因,因为杰克逊没有得到传递类的内部接口的具体类,但我的问题是如何解决它?我该如何反序列化呢?我试图反序列化的类是:

class BaseMetricImpl<N> implements Metric<N> {
    protected MetricValueDescriptor descriptor;
}
Run Code Online (Sandbox Code Playgroud)

MetricValueDescriptor是一个界面,所以这给了我以下错误: -

com.fasterxml.jackson.databind.JsonMappingException: Can not construct instance of MetricValueDescriptor, problem: abstract types either need to be mapped to concrete types, have custom deserializer, or be instantiated with additional type information
 at [Source: java.io.ByteArrayInputStream@2ede2c9f; line: 1, column: 2] (through reference chain: SingleValueMetricImpl["descriptor"])
    at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:164)
    at com.fasterxml.jackson.databind.DeserializationContext.instantiationException(DeserializationContext.java:624)
    at com.fasterxml.jackson.databind.deser.AbstractDeserializer.deserialize(AbstractDeserializer.java:115)
    at com.fasterxml.jackson.databind.deser.SettableBeanProperty.deserialize(SettableBeanProperty.java:375)
    at com.fasterxml.jackson.databind.deser.impl.MethodProperty.deserializeAndSet(MethodProperty.java:98)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:308)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:121)
    at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:2793)
    at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:1989)
Run Code Online (Sandbox Code Playgroud)

Har*_*non 48

杰克逊显然无法构造MetricValueDescriptor对象,因为它是一个接口.您需要在json和ObjectMapper中获得其他信息,以告诉jackson如何构造一个对象.这是一种方法,假设MVDImpl是一个实现的具体类MetricValueDescriptor:

你可以通过json本身的字段告诉杰克逊所需的类型信息"type".为此,您需要在界面中使用JsonTypeInfoJsonSubTypes注释.例如,

@JsonTypeInfo(
    use = JsonTypeInfo.Id.NAME,
    include = JsonTypeInfo.As.PROPERTY,
    property = "type")
@JsonSubTypes({
    @Type(value = MVDImpl.class, name = "mvdimpl") })
interface MetricValueDescriptor
{
   ...
}
Run Code Online (Sandbox Code Playgroud)

您还需要"type":"mvdimpl"在json中添加一个字段.

我将指向正式文档以获取更多信息,但后来我发现了一个很好的博客,涵盖了这个主题 - 与杰森分解JSON.它非常全面地并通过示例介绍了这个主题.所以如果你需要更多的自定义,你一定要阅读它.

  • 我面临同样的问题,但是,"超级"不属于我,不能修改.在那种情况下,我们怎样才能让"杰克逊"地图出现在正确的"子类"中? (2认同)
  • @lvarayut可以使用[Jackson Mixin注释](http://wiki.fasterxml.com/JacksonMixInAnnotations)将注释添加到不可修改的类。 (2认同)

xba*_*esx 13

我看到它采用两种方式之一,但它们都需要您手动创建一个实现接口的具体类。

  1. 使用@Hari Menon 的答案并使用@JsonSubTypes. 如果您可以引入类型字段或其他内容来触发要使用的实现,则此方法有效。
  2. 用于@JsonDeserialize告诉 Jackson 默认使用什么具体类。
@JsonDeserialize(as = MVDImpl.class)
interface MetricValueDescriptor
{
   ...
}
Run Code Online (Sandbox Code Playgroud)

这是更彻底的解释:https://zenidas.wordpress.com/recipes/jackson-deserialization-of-interfaces/

和文档:https://fasterxml.github.io/jackson-databind/javadoc/2.8/com/fasterxml/jackson/databind/annotation/JsonDeserialize.html


All*_*naz 13

您不需要更改代码,可以在映射器上以编程方式设置它:

static setup() {
   final var simpleModule = new SimpleModule()
        .addAbstractTypeMapping(<Interface>.class, <Implementation>.class);

   objMapper = new ObjectMapper()
        .registerModule(new Jdk8Module()) // You probably want this as well
        .registerModule(simpleModule);
}
Run Code Online (Sandbox Code Playgroud)