Я разрабатываю приложение для Android
. Я использую модификацию с Gson
чтобы звонить на сервер и сериализовать/десериализовать. В запросе есть класс ответа, который содержит timestamp
(long
). Иногда это double
, так как iOS
версия приложения использует double
для хранения временных меток.
Есть ли способ заставить Gson
десериализовать ответ и привести его к объекту, присутствующему в классе Response
(long)?
Исключение:
Caused by com.google.gson.JsonSyntaxException: java.lang.NumberFormatException: Expected a long but was 1555323142345.364 at line 1 column 1002 path $.userSettings.stories[4].readTimestamp
at com.google.gson.internal.bind.TypeAdapters$11.read(TypeAdapters.java:323)
at com.google.gson.internal.bind.TypeAdapters$11.read(TypeAdapters.java:313)
at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$1.read(ReflectiveTypeAdapterFactory.java:129)
at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.read(ReflectiveTypeAdapterFactory.java:220)
at com.google.gson.internal.bind.TypeAdapterRuntimeTypeWrapper.read(TypeAdapterRuntimeTypeWrapper.java:41)
at com.google.gson.internal.bind.CollectionTypeAdapterFactory$Adapter.read(CollectionTypeAdapterFactory.java:82)
at com.google.gson.internal.bind.CollectionTypeAdapterFactory$Adapter.read(CollectionTypeAdapterFactory.java:61)
at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$1.read(ReflectiveTypeAdapterFactory.java:129)
at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.read(ReflectiveTypeAdapterFactory.java:220)
at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$1.read(ReflectiveTypeAdapterFactory.java:129)
at com.google.gson.internal.bind.ReflectiveTypeAdapterFactory$Adapter.read(ReflectiveTypeAdapterFactory.java:220)
at retrofit2.converter.gson.GsonResponseBodyConverter.convert(GsonResponseBodyConverter.java:37)
at retrofit2.converter.gson.GsonResponseBodyConverter.convert(GsonResponseBodyConverter.java:25)
at retrofit2.ServiceMethod.toResponse(ServiceMethod.java:119)
at retrofit2.OkHttpCall.parseResponse(OkHttpCall.java:218)
at retrofit2.OkHttpCall.execute(OkHttpCall.java:180)
at retrofit2.ExecutorCallAdapterFactory$ExecutorCallbackCall.execute(ExecutorCallAdapterFactory.java:91)
at inesc_id.pt.motivandroid.motviAPIClient.MotivAPIClientManager$CheckOnboardingNeededAndRetrieve.doInBackground(MotivAPIClientManager.java:909)
at inesc_id.pt.motivandroid.motviAPIClient.MotivAPIClientManager$CheckOnboardingNeededAndRetrieve.doInBackground(MotivAPIClientManager.java:711)
at android.os.AsyncTask$2.call(AsyncTask.java:307)
at java.util.concurrent.FutureTask.run(FutureTask.java:237)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1113)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:588)
at java.lang.Thread.run(Thread.java:833)
Учебный класс:
public class StoryStateful implements Serializable{
@Expose
int storyID;
@Expose
boolean read;
@Expose
long readTimestamp;
@Expose
long availableTimestamp;
}
Для простоты предположим, что ваша полезная нагрузка JSON
:
{
"timestamp": 1555323142345.345
}
И это должно соответствовать:
class Pojo {
private long timestamp;
// getters, setters, toString
}
Решение зависит от того, можете ли вы изменить модель Pojo
или нет. Если да и вы хотите хранить double
и long
можно изменить тип на Number
:
private Number timestamp;
И вы должны быть в состоянии разобрать double
и long
метки времени. Если вы всегда хотите long
вам нужно реализовать собственный десериализатор:
class LongOrDoubleJsonDeserializer implements JsonDeserializer<Long> {
@Override
public Long deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
if (json.isJsonPrimitive()) {
Number number = json.getAsNumber();
return number.longValue();
}
return null;
}
}
И ваше свойство Pojo
должно выглядеть так:
@JsonAdapter(LongOrDoubleJsonDeserializer.class)
private long timestamp;
Если вы не можете изменить класс Pojo
вам нужно реализовать собственный TypeAdapterFactory
и зарегистрироваться, используя GsonBuilder
. Смотрите эту ссылку: Создание экземпляра Retrofit. Пользовательская реализация может выглядеть так:
class LongOrDoubleTypeAdapterFactory implements TypeAdapterFactory {
@Override
public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> type) {
Class<? super T> rawType = type.getRawType();
if (rawType == Long.class || rawType == long.class) {
return new TypeAdapter<T>() {
@Override
public void write(JsonWriter out, T value) {
}
@Override
public T read(JsonReader in) throws IOException {
try {
return (T) new Long(in.nextLong());
} catch (NumberFormatException e) {
return (T) new Long(((Double) in.nextDouble()).longValue());
}
}
};
}
return null;
}
}
И зарегистрируйтесь как ниже:
Gson gson = new GsonBuilder()
.registerTypeAdapterFactory(new LongOrDoubleTypeAdapterFactory())
.create();