Как десериализовать поля интерфейса с помощью метода 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 с Джексоном. Он охватывает эту тему довольно подробно и с примерами. Поэтому вам обязательно нужно прочитать его, если вам нужна дополнительная настройка.