杰克逊序列化抽象类

12
我正在尝试使用JSON ObjectMapper反序列化对象,但在尝试反序列化时出现以下错误。
com.fasterxml.jackson.databind.JsonMappingException: 无法构建 com.phoenix.types.OrderItem 的实例:抽象类型需要映射到具体类型,具有自定义反序列化器,或包含附加类型信息 [来源:java.io.StringReader@4bb33f74; 行:112,列:7](通过引用链:com.phoenix.types.GenerateRequest ["order"] - >com.phoenix.types.Order ["orderItems"] - > Object [] [0]) 在 com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:261) 在 com.fasterxml.jackson.databind.DeserializationContext.instantiationException(DeserializationContext.java:1456) 在 com.fasterxml.jackson.databind.DeserializationContext.handleMissingInstantiator(DeserializationContext.java:1012) 在 com.fasterxml.jackson.databind.deser.AbstractDeserializer.deserialize(AbstractDeserializer.java:149) 在 com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.deserialize(ObjectArrayDeserializer.java:196) 在 com.fasterxml.jackson.databind.deser.std.ObjectArrayDeserializer.deserialize(ObjectArrayDeserializer.java:20) 在 com.fasterxml.jackson.databind.deser.SettableBeanProperty.deserialize(SettableBeanProperty.java:499) 在 com.fasterxml.jackson.databind.deser.impl.MethodProperty.deserializeAndSet(MethodProperty.java:101) 在 com.fasterxml.jackson.databind.deser.BeanDeserializer.vanillaDeserialize(BeanDeserializer.java:276) 在 com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:140) 在 com.fasterxml.jackson.databind.deser.SettableBeanProperty.deserialize(SettableBeanProperty.java:499) 在 com.fasterxml.jackson.databind.deser.impl.MethodProperty.deserializeAndSet(MethodProperty.java:101) 在 com.fasterxml.jackson.databind.deser.BeanDeserializer.vanillaDeserialize(BeanDeserializer.java:276) 在 com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:140) 在 com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:3789) 在 com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:2833)

我遇到了一个文章, 介绍了多态反序列化的解决方案。这基本上提供了解决上述错误的方法。用于反序列化的类(在此例中为OrderItem等)是jar文件的一部分。然而,是否有一种方法可以在尝试进行反序列化时将JsonDeserialize定义为objectmapper的一部分,而不是向类本身添加注释,因为我无法访问它。

3个回答

18

是的,你可以为这个抽象类编写自己的自定义反序列化器。该反序列化器需要确定JSON表示哪个具体类,并实例化一个该类的实例。

可能有更符合惯用法的方法,但以下是一个快速而简单的示例:

public class Test {
    public static void main(String... args) throws IOException {
        final ObjectMapper mapper = new ObjectMapper();
        final SimpleModule module = new SimpleModule();
        module.addDeserializer(Animal.class, new AnimalDeserializer());
        mapper.registerModule(module);

        final String json = "{\"aGoodBoy\": true}";
        final Animal animal = mapper.readValue(json, Animal.class);
        System.out.println(animal.talk());
    }

    public static abstract class Animal {
        public abstract String talk();
    }

    public static class Fish extends Animal {
        @Override
        public String talk() {
            return "blub blub I'm a dumb fish";
        }
    }

    public static class Dog extends Animal {
        public boolean aGoodBoy;

        @Override
        public String talk() {
            return "I am a " + (aGoodBoy ? "good" : "bad") + " dog";
        }
    }

    public static class AnimalDeserializer extends StdDeserializer<Animal> {
        protected AnimalDeserializer() {
            this(null);
        }

        protected AnimalDeserializer(final Class<?> vc) {
            super(vc);
        }

        @Override
        public Animal deserialize(final JsonParser parser, final DeserializationContext context)
        throws IOException, JsonProcessingException {
            final JsonNode node = parser.getCodec().readTree(parser);
            final ObjectMapper mapper = (ObjectMapper)parser.getCodec();
            if (node.has("aGoodBoy")) {
                return mapper.treeToValue(node, Dog.class);
            } else {
                return mapper.treeToValue(node, Fish.class);
            }
        }
    }
}

谢谢Andrew。有没有示例可以使用自定义反序列化程序指定抽象类的具体类? - Punter Vicky
更新了一个代码示例。我相信有一种更好的方法可以获取底层的ObjectMapper而不是强制转换ObjectCodec,但是我在脑海中找到它有些困难。你也可以手动构造具体的实例,但我想避免这样做的工作。 - Andrew Rueckert
这个救了我的一天。 - आनंद
我赞赏你使用 final。我不明白为什么人们只将其用于“常量”。除非无法,否则每个变量都应该是 final(在我看来)。 - MuffinTheMan

1

-1
Jackson 尝试创建您的抽象类的实例。
在您的抽象类中为此字段编写 @JsonIgnore 注释,并创建此字段的 getter。然后在此 getter 上编写另一个注释 @JsonProperty("{{nameOfYourField}}")。
@JsonIgnore
@JoinColumn(name = "column_name")
private OrderItem orderItem;

@JsonProperty("orderItem")
public OrderItem getOrderItem () {
    return orderItem;
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接