Например, у нас чередуются данные
{
"orderID": 12345,
"shopperName": "Ваня Иванов",
"shopperEmail": "ivanov@example.com",
"contents": [
{
"type": "Image",
"Url": "https://someurl.com",
},
{
"type": "Text",
"text": "some text",
"isItalic": true,
"isBold": true
}
],
"orderCompleted": true
}
Можно ли представить contents как массив HashMap'ов? Допустим, у меня есть 2 класса: Image и Text, как мне определить к какому классу принадлежит элемент в массиве contents?
public class RawCollectionsExample {
static class Event {
private String name;
private String source;
private Event(String name, String source) {
this.name = name;
this.source = source;
}
@Override
public String toString() {
return String.format("(name=%s, source=%s)", name, source);
}
}
@SuppressWarnings({ "unchecked", "rawtypes" })
public static void main(String[] args) {
Gson gson = new Gson();
Collection collection = new ArrayList();
collection.add("hello");
collection.add(5);
collection.add(new Event("GREETINGS", "guest"));
String json = gson.toJson(collection);
System.out.println("Using Gson.toJson() on a raw collection: " + json);
JsonParser parser = new JsonParser();
JsonArray array = parser.parse(json).getAsJsonArray();
String message = gson.fromJson(array.get(0), String.class);
int number = gson.fromJson(array.get(1), int.class);
Event event = gson.fromJson(array.get(2), Event.class);
System.out.printf("Using Gson.fromJson() to get: %s, %d, %s", message, number, event);
}
}
В данном примере мы точно знаем экземпляром какого класса является элемент в массиве
Первый вариант - вы можете создать класс, включающий как свойства Image
, так и свойства Text
и, в последствии, различать их по свойству type
Второй вариант - вы можете определить собственный десериализатор.
Ссылка на джавадоки
Можно реализовать с помощью класса RuntimeTypeAdapterFactory, сразу получив нужные pojo. Класс отсутствует в gson'е, надо скопировать локально. Будет что-то типа:
class Order {
@SerializedName("contents") List<Content> contents;
}
class Content {
}
class ImageContent extends Content {
@SerializedName("Url") private String url;
}
class TextContent extends Content {
@SerializedName("text") String text;
@SerializedName("isItalic") boolean italic;
@SerializedName("isBold") boolean bold;
}
Order getOrder(String jsonString){
RuntimeTypeAdapterFactory<Content> contentAdapterFactory = RuntimeTypeAdapterFactory.of(Content.class, "type")
.registerSubtype(ImageContent.class, "Image")
.registerSubtype(TextContent.class, "Text");
Gson gson = new GsonBuilder().registerTypeAdapterFactory(contentAdapterFactory).create();
return gson.fromJson(jsonString, Order.class);
}
NB В классах нет поля type, оно присутствует только в json'е, как при сериализации, так нужно для десериализации.
ЗЫ Есть баг, там же решение. Без него не работает как надо.
Кофе для программистов: как напиток влияет на продуктивность кодеров?
Рекламные вывески: как привлечь внимание и увеличить продажи
Стратегії та тренди в SMM - Технології, що формують майбутнє сьогодні
Выделенный сервер, что это, для чего нужен и какие характеристики важны?
Современные решения для бизнеса: как облачные и виртуальные технологии меняют рынок
Можно ли использовать текст вместо цифр в NumberPicker ? Если нет, есть ли аналоги NumberPicker с текстом? Заранее спасибо
После успешной авторизации приложения падает, и не выдает никаких ошибокActivity