Полиморф c десериализация Джексона по typeinfo не работает с перечислениями - PullRequest
0 голосов
/ 24 марта 2020

У меня есть следующий класс, являющийся представлением java объекта JSON:

public class Vehicle
{
   private VehicleType vehicleType;

   private VehicleConfiguration vehicleConfiguration;

   @JsonCreator
   public Vehicle( @JsonProperty("vehicleType")final VehicleType vehicleType,
         @JsonProperty("vehicleConfiguration")final VehicleConfiguration vehicleConfiguration )
   {
      this.vehicleType = vehicleType;
      this.vehicleConfiguration = vehicleConfiguration;
   }

   ...

   @JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.EXTERNAL_PROPERTY, property = "vehicleType")
   @JsonSubTypes({ @JsonSubTypes.Type(value = TruckConfiguration.class, name = "TRUCK"),
         @JsonSubTypes.Type(value = CarConfiguration.class, name = "CAR") })
   public void setVehicleConfiguration(
         final VehicleConfiguration vehicleConfiguration )
   {
      this.vehicleConfiguration = vehicleConfiguration;
   }
}

Я использую перечисление vehicleType (имеющее два значения TRUCK и CAR) в качестве внешнего свойства для сообщите десериализатору, как я хочу десериализовать абстрактный класс VehicleConfiguration: либо в CarConfiguration, либо в TruckConfiguration. Вот другие классы (исключая шаблон):

public class CarConfiguration extends VehicleConfiguration
{
   private int numberOfDoors;


   @JsonCreator
   public CarConfiguration( @JsonProperty("numberOfDoors") final int numberOfDoors )
   {
      this.numberOfDoors = numberOfDoors;
   }
}
public class TruckConfiguration extends VehicleConfiguration
{
   private String manufacturer;


   @JsonCreator
   public TruckConfiguration( @JsonProperty("manufacturer") final String manufacturer )
   {
      this.manufacturer = manufacturer;
   }
}

Мой тест следующий:

@Test
   public void given_when_then() throws IOException
   {
      // GIVEN
      final String json = "{\n"
            + "    \"vehicleType\": \"TRUCK\",\n"
            + "    \"vehicleConfiguration\": {\n"
            + "      \"manufacturer\": \"SCANIA\"\n"
            + "    }\n"
            + "}";

      // WHEN
      ObjectMapper mapper = new ObjectMapper();
      mapper.enableDefaultTyping();

      Vehicle vehicle = mapper.readValue(json, Vehicle.class);

      // THEN
      assertEquals((( TruckConfiguration ) vehicle.getVehicleConfiguration()).getManufacturer(), "SCANIA");
   }

Выдает следующее исключение:

com.fasterxml.jackson.databind.JsonMappingException: Can not construct instance of com.test.Vehicle, problem: argument type mismatch
 at [Source: {
    "vehicleType": "TRUCK",
    "vehicleConfiguration": {
      "manufacturer": "SCANIA"
    }
}; line: 6, column: 1]

    at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:277)
    at com.fasterxml.jackson.databind.DeserializationContext.instantiationException(DeserializationContext.java:1441)
    at com.fasterxml.jackson.databind.deser.std.StdValueInstantiator.wrapAsJsonMappingException(StdValueInstantiator.java:476)
    at com.fasterxml.jackson.databind.deser.std.StdValueInstantiator.rewrapCtorProblem(StdValueInstantiator.java:495)
    at com.fasterxml.jackson.databind.deser.std.StdValueInstantiator.createFromObjectWith(StdValueInstantiator.java:276)
    at com.fasterxml.jackson.databind.deser.ValueInstantiator.createFromObjectWith(ValueInstantiator.java:228)
    at com.fasterxml.jackson.databind.deser.impl.PropertyBasedCreator.build(PropertyBasedCreator.java:135)
    at com.fasterxml.jackson.databind.deser.impl.ExternalTypeHandler.complete(ExternalTypeHandler.java:230)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeUsingPropertyBasedWithExternalTypeId(BeanDeserializer.java:937)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeWithExternalTypeId(BeanDeserializer.java:792)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:312)
    at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:148)
    at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:3798)
    at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:2842)
    at com.test.VehicleTest.given_when_then(VehicleTest.java:44)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:47)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:44)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:271)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:70)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:238)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:63)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:236)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:53)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:229)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:309)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:160)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
    at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
    at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
    at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)
Caused by: java.lang.IllegalArgumentException: argument type mismatch
    at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
    at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62)
    at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
    at java.lang.reflect.Constructor.newInstance(Constructor.java:423)
    at com.fasterxml.jackson.databind.introspect.AnnotatedConstructor.call(AnnotatedConstructor.java:124)
    at com.fasterxml.jackson.databind.deser.std.StdValueInstantiator.createFromObjectWith(StdValueInstantiator.java:274)
    ... 32 more

Кажется, что проблема в enum vehicleType как typeInfo: если я изменяю это поле на простую строку и запускаю тот же тест, он работает без проблем. Но в моем проекте у меня уже есть перечисление, которое мне нужно использовать, чтобы дифференцировать десериализацию.

Мое временное решение состояло в том, чтобы добавить поле vehicleType в абстрактную VehicleConfiguration и установить аннотации непосредственно в абстрактном классе, используя аннотацию JsonTypeInfo.As.PROPERTY. Таким образом, он работает даже с перечислением, но я добавляю избыточную информацию в поле vehicleType. Это ошибка десериализатора или я что-то упустил?

1 Ответ

0 голосов
/ 26 марта 2020

После долгих поисков и поисков, удалось найти решение: удалил конструктор, и теперь все работает просто отлично.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...