4時間ノンストップで問題を解決しようとした後、誰かが私を助けてくれるかどうかここで尋ねることにしました。
問題は、サーバーから受信したデータを逆シリアル化しようとすると、Androidクライアントが「Unparseable:1302828677828」例外をスローすることです。
Gsonを使用してミリ秒形式の日付を逆シリアル化できるかどうか知りたいのですが。
4時間ノンストップで問題を解決しようとした後、誰かが私を助けてくれるかどうかここで尋ねることにしました。
問題は、サーバーから受信したデータを逆シリアル化しようとすると、Androidクライアントが「Unparseable:1302828677828」例外をスローすることです。
Gsonを使用してミリ秒形式の日付を逆シリアル化できるかどうか知りたいのですが。
long
し、その後、プログラムの変換long
にDate
あなたのコードでは?
回答:
アルフォンソのコメント:
ついに私は解決策を得ました:
// Creates the json object which will manage the information received GsonBuilder builder = new GsonBuilder(); // Register an adapter to manage the date types as long values builder.registerTypeAdapter(Date.class, new JsonDeserializer<Date>() { public Date deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException { return new Date(json.getAsJsonPrimitive().getAsLong()); } }); Gson gson = builder.create();
.getAsJsonPrimitive()
省略、およびJava 8ラムダでそれも短いことができます:Gson gson = new GsonBuilder().registerTypeAdapter(Date.class, (JsonDeserializer) (json, typeOfT, context) -> new Date(json.getAsLong())).create();
デフォルトの日付形式とタイムスタンプ(長い)形式をサポートするGSONのデフォルトのDateTypeAdapterに基づいてImprovedDateTypeAdapterを作成しました。
import com.google.gson.Gson;
import com.google.gson.JsonSyntaxException;
import com.google.gson.TypeAdapter;
import com.google.gson.TypeAdapterFactory;
import com.google.gson.reflect.TypeToken;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.JsonToken;
import com.google.gson.stream.JsonWriter;
import java.io.IOException;
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
import java.util.TimeZone;
public final class ImprovedDateTypeAdapter extends TypeAdapter<Date> {
public static final TypeAdapterFactory FACTORY = new TypeAdapterFactory() {
public <T> TypeAdapter<T> create(Gson gson, TypeToken<T> typeToken) {
@SuppressWarnings("unchecked")
TypeAdapter<T> typeAdapter = (TypeAdapter<T>) ((typeToken.getRawType() == Date.class) ? new ImprovedDateTypeAdapter()
: null);
return typeAdapter;
}
};
private final DateFormat enUsFormat;
private final DateFormat localFormat;
private final DateFormat iso8601Format;
public ImprovedDateTypeAdapter() {
this.enUsFormat = DateFormat.getDateTimeInstance(2, 2, Locale.US);
this.localFormat = DateFormat.getDateTimeInstance(2, 2);
this.iso8601Format = buildIso8601Format();
}
private static DateFormat buildIso8601Format() {
DateFormat iso8601Format = new SimpleDateFormat(
"yyyy-MM-dd'T'HH:mm:ss'Z'", Locale.US);
iso8601Format.setTimeZone(TimeZone.getTimeZone("UTC"));
return iso8601Format;
}
public Date read(JsonReader in) throws IOException {
if (in.peek() == JsonToken.NULL) {
in.nextNull();
return null;
}
return deserializeToDate(in.nextString());
}
private synchronized Date deserializeToDate(String json) {
try {
return new Date(Long.parseLong(json));
} catch (Exception e) {
try {
return this.localFormat.parse(json);
} catch (ParseException e1) {
try {
return this.enUsFormat.parse(json);
} catch (ParseException e2) {
try {
return this.iso8601Format.parse(json);
} catch (ParseException e3) {
throw new JsonSyntaxException(json, e3);
}
}
}
}
}
public synchronized void write(JsonWriter out, Date value)
throws IOException {
if (value == null) {
out.nullValue();
return;
}
String dateFormatAsString = this.enUsFormat.format(value);
out.value(dateFormatAsString);
}
}
それを使用するには:
// Creates the json object which will manage the information received
GsonBuilder builder = new GsonBuilder();
// Register an adapter to manage the date types as long values
builder.registerTypeAdapter(Date.class, new ImprovedDateTypeAdapter());
Gson gson = builder.create();
JsonSerializer<Date> serializer= new JsonSerializer<Date>() {
@Override
public JsonElement serialize(Date src, Type typeOfSrc, JsonSerializationContext
context) {
return src == null ? null : new JsonPrimitive(src.getTime());
}
};
JsonDeserializer<Date> deserializer= new JsonDeserializer<Date>() {
@Override
public Date deserialize(JsonElement json, Type typeOfT,
JsonDeserializationContext context) throws JsonParseException {
return json == null ? null : new Date(json.getAsLong());
}
};
Gson gson = new GsonBuilder()
.registerTypeAdapter(Date.class, serializer)
.registerTypeAdapter(Date.class, deserializer).create();
JSONの処理中にミリ秒を日付に変換するには、以下のスニペットを使用してください。
GsonBuilder gsonBuilder = new GsonBuilder();
// Adapter to convert long values to date types
gsonBuilder.registerTypeAdapter(Date.class, new JsonDeserializer<Date>() {
public Date deserialize(JsonElement jsonElement, Type typeOfObj, JsonDeserializationContext context)
throws JsonParseException {
//Converting milliseconds to current Date. (instead of 1970)
return new Date(jsonElement.getAsLong() * 1000);
}
});
Gson gson = gsonBuilder.setPrettyPrinting().create();
AndroidアノテーションライブラリのRESTクライアントでDateTimeフィールドを逆シリアル化しようとしたときに同じ問題が発生します。解決策として、カスタムGsonHttpMessageConverterを作成しました
public class CustomGsonHttpMessageConverter extends GsonHttpMessageConverter {
public CustomGsonHttpMessageConverter() {
// Creates the json object which will manage the information received
GsonBuilder builder = new GsonBuilder();
// Register an adapter to manage the date types as long values
builder.registerTypeAdapter(Date.class, new JsonDeserializer<Date>() {
public Date deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
return new Date(json.getAsJsonPrimitive().getAsLong());
}
});
setGson(builder.create());
}
}
RESTクライアントで定義します
@Rest(rootUrl = "http://192.168.1.1:8080", converters = {CustomGsonHttpMessageConverter.class})
public interface RestClient extends RestClientErrorHandling {
...
お役に立てば幸いです
何らかの理由で、匿名クラスを使用した上記のコードでIntellijでコンパイルエラーが発生しました。ラムダは私のために働いた:
private static Gson buildGson(){
// Deserialize longs as Dates
final JsonDeserializer<Date> dateDeserializer = (json, type, context) -> json == null ? null : new Date(json.getAsLong());
return new GsonBuilder().registerTypeAdapter(Date.class, dateDeserializer).create();
}