JsonSubTypes, список полиморфных объектов и Parcelable

Моя структура JSON:

{  
     ...
     "type":"post", // Type could vary
     "items":[]     // Array of items, each item is typeOf("type") 
     ...
}  

Как я могу десериализовать и правильно поместить пакет items внутри моего POJO:

public class ItemsEnvelope {
    private String type;

    @JsonTypeInfo(
            use = JsonTypeInfo.Id.NAME,
            include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
            property = "type",
            visible = true)
    @JsonSubTypes({
            @JsonSubTypes.Type(value = A.class, name = "A"),
            @JsonSubTypes.Type(value = B.class, name = "B"),
            @JsonSubTypes.Type(value = C.class, name = "C")
    })
    private List<Item> items;

    interface Item extends Parcelable {}

    class A implements Item {
        // Bunch of getters/setters and Parcelable methods/constructor
    }

    class B implements Item {
        // Bunch of getters/setters and Parcelable methods/constructor
    }

    class C implements Item {
        // Bunch of getters/setters and Parcelable methods/constructor
    }

    // Bunch of getters/setters and Parcelable methods/constructor
}

В список набранных пакетов должен быть предоставлен объект CREATOR, который, очевидно, не может иметь интерфейс. Должен ли я использовать абстрактный класс вместо интерфейса?

Ответ 1

Хорошо, так как Джексон ожидает информацию о типе в каждом элементе списка, и я не хотел писать пользовательский десериализатор для этого POJO - я решил это другим способом.

Прежде всего, я сделал интерфейс, который должен был реализовать все мои элементы подтипа.

@JsonTypeInfo(
        use = JsonTypeInfo.Id.NAME,
        include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
        property = "type",
        visible = true)
@JsonSubTypes({
        @JsonSubTypes.Type(value = PostFeedItem.class, name = BaseFeedItem.TYPE_POST),
        @JsonSubTypes.Type(value = PhotoFeedItem.class, name = BaseFeedItem.TYPE_PHOTO),
        @JsonSubTypes.Type(value = AudioFeedItem.class, name = BaseFeedItem.TYPE_AUDIO),
        @JsonSubTypes.Type(value = VideoFeedItem.class, name = BaseFeedItem.TYPE_VIDEO),
        @JsonSubTypes.Type(value = FriendFeedItem.class, name = BaseFeedItem.TYPE_FRIEND)
})
public interface FeedItem extends Parcelable {
    // ...
    @BaseFeedItem.Type
    String getType();
    // ...
}

Затем я создал базовый класс, который должен был расширять все мои элементы подтипа.

public abstract class BaseFeedItem implements FeedItem {
    public static final String TYPE_POST = "post";
    public static final String TYPE_COMMUNITY_POST = "group_post";
    public static final String TYPE_PHOTO = "photo";
    public static final String TYPE_AUDIO = "audio";
    public static final String TYPE_VIDEO = "video";
    public static final String TYPE_FRIEND = "friend";

    @Retention(RetentionPolicy.SOURCE)
    @StringDef({TYPE_POST, TYPE_COMMUNITY_POST, TYPE_PHOTO, TYPE_AUDIO, TYPE_VIDEO, TYPE_FRIEND})
    public @interface Type {}
    private String type;

    @Type
    public String getType() {
        return type;
    }
    // ...
}

Наконец, мой класс POJO:

public class NewsFeedEnvelope {
    // ...
    @JsonProperty("rows")
    private List<FeedItem> items;
    // ...
}

Теперь POJO успешно автоматически десериализуется Jackson без каких-либо пользовательских десериализаторов.

Ответ 2

Прежде всего: EXTERNAL_PROPERTY будет работать только тогда, когда значение заключено в другой объект (POJO); и не будет работать для List s, массивов или Map s.

Если вы использовали один из других методов включения, все должно работать так, что вы можете сериализовать контент как JSON и читать готовые JSON-поддерживающие типы, как ожидалось. То есть, сериализация в оба конца поддерживается при запуске с объектов Java. Помимо этого могут поддерживаться некоторые структуры кодирования JSON; но это зависит от конкретной структуры, которую вы пытаетесь поддерживать.