Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Jackson polymorphism: How to map multiple subtypes to the same class

Tags:

java

json

jackson

I'm using Jackson 1.9.x. Sticking with the Animals example, Here's what I'd like to do:

Let's say I have an Animal class:

public class Animal {
    private String type;
    // accessors
}

public class Mammal extends Animal {
    private String diet;
    // accessors
}

public class Bird extends Animal {
    private boolean tropical;
    // accessors
}

I would like to be able to do something like this (where I map a few subtypes to one class, and a few more to a different class):

@JsonTypeInfo(use = Id.NAME, include = As.PROPERTY, property = "type")
@JsonSubTypes({@JsonSubTypes.Type(value = Mammal.class, name = "Dog"),
                @JsonSubTypes.Type(value = Mammal.class, name = "Cat"),
                @JsonSubTypes.Type(value = Bird.class, name = "Dodo"},
                @JsonSubTypes.Type(value = Bird.class, name = "Cockatoo"})
public class Animal {

}

What I'm seeing right now is that Jackson will only recognize the Dog-to-Mammal and the Dodo-to-Bird mapping. This is because StdSubtypeResolver._collectAndResolve() only allows the same class to get registered once (due to the implementation of NamedType.equals()).

Is there a workaround to the issue I'm seeing?

like image 432
Yahya Cahyadi Avatar asked Aug 23 '13 04:08

Yahya Cahyadi


3 Answers

I also faced the same issue and found out that the Subtype mapping expects unique classes.

What I did was to create two classes that extend the same base class. The extended classes are empty as they have the same properties as base class. Then added them to the Subtype map. For example, in your case, it will be -

@JsonTypeInfo(use = Id.NAME, include = As.PROPERTY, property = "type")
@JsonSubTypes({@JsonSubTypes.Type(value = Mammal.class, name = "Dog"),
            @JsonSubTypes.Type(value = Mammal.class, name = "Cat"),
            @JsonSubTypes.Type(value = BirdDodo.class, name = "Dodo"},
            @JsonSubTypes.Type(value = BirdCockatoo.class, name = "Cockatoo"})
public class Animal {

}

public class BirdCockatoo extends Cockatoo{}
public class BirdDodo extends Dodo{}

I understand it is the not the best approach but until the issue is not resolved, it could be the best way to fix this. I followed this approach for now.

Hope it helps you!

like image 72
hkasera Avatar answered Nov 11 '22 12:11

hkasera


The bug has been resolved in the version 2.6.0, so you just have to update Jackson to version 2.6.0 or later. The additional information is here and here.

like image 7
erkfel Avatar answered Nov 11 '22 13:11

erkfel


Perhaps not by using annotations. Problems comes from the fact that such mapping would not work for serialization, and existing mapping does expect one-to-one (bijection) relationship. But you may want to file an RFE at jackson-databind issue tracker; adding support may be possible.

like image 4
StaxMan Avatar answered Nov 11 '22 11:11

StaxMan