Десериализация Gson списка<String> в realmList<RealmString>

Я использую модификацию с помощью gson, чтобы десериализовать мой json в объекты области. Это работает очень хорошо по большей части. Проблема возникает при работе с

RealmList(String(или любой другой базовый тип данных))

Поскольку Realm не поддерживает RealmList, где E не расширяет объект Realm, я обернул String в RealmObject.

public class RealmString extends RealmObject {
  private String val;

  public String getValue() {
    return val;
  }

  public void setValue(String value) {
    this.val = value;
  }
}

Мой предмет Царства, как показано ниже

    public class RealmPerson extends RealmObject {
    @PrimaryKey
    private String userId;
    ...
    private RealmList<RealmString> stringStuff;
    private RealmList<SimpleRealmObj> otherStuff;

    <setters and getters>
   }

SimpleRealmObj работает нормально, так как содержит только элементы String

    public class SimpleRealmObj extends RealmObject {
    private String foo;
    private String bar;
       ...
    }

Как я могу десериализовать stringStuff? Я пытался использовать gson TypeAdapter

public class RealmPersonAdapter extends TypeAdapter<RealmPerson> {
    @Override
    public void write(JsonWriter out, RealmPerson value) throws IOException {
        out.beginObject();
        Log.e("DBG " + value.getLastName(), "");
        out.endObject();
    }

    @Override
    public RealmPerson read(JsonReader in) throws IOException {
        QLRealmPerson rList = new RealmPerson();
        in.beginObject();
        while (in.hasNext()) {
            Log.e("DBG " + in.nextString(), "");
        }
        in.endObject();

        return rList;
    }

Однако я все еще ударил IllegalStateException

2334-2334 / com.qualcomm.qlearn.app E // PersonService.java: 71 ﹕ main com.google.gson.JsonSyntaxException: java.lang.IllegalStateException: Ожидается строка, но она ИМЯ в строке 1 пути 3 столбца $.

Я пробовал RealmList, адаптер RealmString ранее, но безрезультатно. Единственный обходной путь, который мне удалось найти, - это https://github.com/realm/realm-java/issues/620 Есть лучшие варианты?

4 ответа

Решение

Сообщение об ошибкеExpected a string but was NAME"может быть решена путем получения имени объекта json в JsonReader перед фактическим объектом JSON (который является String в твоем случае).

Вы можете взглянуть на документацию по Android для JsonReader. Имеет подробное объяснение и фрагмент кода. Вы также можете взглянуть на readMessage метод в примере кода кода в документации.

Я изменил ваш read метод к тому, что я думаю, должно быть. ПРИМЕЧАНИЕ. Я не тестировал код, поэтому в нем могут быть незначительные ошибки.

@Override
public RealmPerson read(JsonReader in) throws IOException {
    RealmPerson rList = new RealmPerson();
    in.beginObject();
    String name = "";
    while (in.hasNext()) {
        name = in.nextName();

        if (name.equals("userId")) {
            String userId = in.nextString();
            // update rList here 
        } else if (name.equals("otherStuff")) {
            // since otherStuff is a RealmList of RealmStrings,
            // your json data would be an array
            // You would need to loop through the array to retrieve 
            // the json objects
            in.beginArray();
            while (in.hasNext()) {
                // begin each object in the array
                in.beginObject();
                name = in.nextName();
                // the RealmString object has just one property called "value"
                // (according to the code snippet in your question)
                if (name.equals("val")) {
                    String val = in.nextString();
                     // update rList here 
                } else {
                    in.skipValue();
                }
                in.endObject();
            }
            in.endArray();
        } else {
            in.skipValue();
        }
    }
    in.endObject();


    return rList;
}

Позвольте мне знать, если это помогает.

Лучше использовать JsonSerializer а также JsonDeserializer скорее, чем TypeAdapter для тебя RealmObject по двум причинам:

  1. Они позволяют вам делегировать (де) сериализацию для вашего RealmObject к стандартному сериализатору Gson (de), что означает, что вам не нужно писать шаблон самостоятельно.

  2. В Gson 2.3.1 есть странная ошибка, которая может вызвать StackruError во время десериализации (я попробовал TypeAdapter Подойди сам и столкнулся с этой ошибкой).

Вот как (заменить Tag с вашим RealmObject учебный класс):

(Обратите внимание, что context.serialize а также context.deserialize ниже эквивалентны gson.toJson а также gson.fromJson Это означает, что нам не нужно разбирать Tag класс сами.)

Парсер + сериализатор для RealmList<Tag>:

public class TagRealmListConverter implements JsonSerializer<RealmList<Tag>>,
        JsonDeserializer<RealmList<Tag>> {

    @Override
    public JsonElement serialize(RealmList<Tag> src, Type typeOfSrc,
                                 JsonSerializationContext context) {
        JsonArray ja = new JsonArray();
        for (Tag tag : src) {
            ja.add(context.serialize(tag));
        }
        return ja;
    }

    @Override
    public RealmList<Tag> deserialize(JsonElement json, Type typeOfT,
                                      JsonDeserializationContext context)
            throws JsonParseException {
        RealmList<Tag> tags = new RealmList<>();
        JsonArray ja = json.getAsJsonArray();
        for (JsonElement je : ja) {
            tags.add((Tag) context.deserialize(je, Tag.class));
        }
        return tags;
    }

}

Класс тегов:

@RealmClass
public class Tag extends RealmObject {
    private String value;

    public String getValue() {
        return value;
    }

    public void setValue(String value) {
        this.value = value;
    }
}

Затем зарегистрируйте свой класс конвертера в Gson:

Gson gson = new GsonBuilder()
        .registerTypeAdapter(new TypeToken<RealmList<Tag>>() {}.getType(),
                new TagRealmListConverter())
        .create();

Мой gson typeAdapter был виновником. Вышеупомянутая ошибка была замечена, поскольку я неправильно десериализовал json в RealmPerson, первое поле не является строкой, следовательно,

in.nextString()

был разорением

Я посмотрел на пример кода, и он ударил меня, мне не пришлось использовать

in.beginObject() и in.endObject()

десериализовать строку. Код ниже работает.

public class QLRealmStringAdapter extends TypeAdapter<QLRealmString> {
@Override
public void write(JsonWriter out, QLRealmString value) throws IOException {
    Log.e("DBG " + value.getValue(), "");
    out.value(value.getValue());
}

@Override
public RealmString read(JsonReader in) throws IOException {
    RealmString rString = new RealmString();
    if (in.hasNext()) {
        String nextStr = in.nextString();
        System.out.println("DBG " + nextStr);
        rString.setValue(nextStr);
    }

    return rString;
}

}

Надеюсь, это кому-нибудь поможет.

Мне нужен сериализатор и десериализатор Джексона для преобразования Arraylist в RealmList

Другие вопросы по тегам