Как десериализовать поля интерфейса с помощью метода Jackson objectMapper?

Для функции

ObjectMapper readValue(InputStream in, Class<T> valueType) требуется класс. Но как я могу использовать его, если класс, который я передаю внутри, имеет некоторый интерфейс в качестве члена данных.

хотя я могу понять причину этого исключения, поскольку Джексон не получает конкретный класс внутреннего интерфейса пройденного класса, но мой вопрос - как его решить? как мне его десериализовать? Класс, который я пытаюсь десериализовать, следующий:

class BaseMetricImpl<N> implements Metric<N> {
    protected MetricValueDescriptor descriptor;
}

Здесь 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: [email protected]; 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)

Ответ 1

Джексон, очевидно, не может построить объект MetricValueDescriptor, так как это интерфейс. Вам нужно будет получить дополнительную информацию в вашем json и в ObjectMapper, чтобы сообщить джексону, как построить объект из него. Вот один из способов сделать это, предполагая, что MVDImpl - это конкретный класс, который реализует MetricValueDescriptor:

Вы можете сообщить Джексону требуемую информацию о типе через поле в самом json, скажем "type". Для этого вам нужно использовать JsonTypeInfo и JsonSubTypes аннотации в вашем интерфейсе. Например,

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

Вам также нужно добавить поле "type":"mvdimpl" в json.

Я собирался указать вам на официальный документ для получения дополнительной информации, но затем я нашел отличный блог, посвященный этой теме - Deserialize JSON с Джексоном. Он охватывает эту тему довольно подробно и с примерами. Поэтому вам обязательно нужно прочитать его, если вам нужна дополнительная настройка.