REST呼び出しにRetrofitライブラリを使用しています。私がやったことのほとんどはバターとしてスムーズでしたが、何らかの理由でJSONタイムスタンプ文字列をJava.util.Date
オブジェクトに変換する際に問題が発生しています。入ってくるJSONは次のようになります。
{
"date": "2013-07-16",
"created_at": "2013-07-16T22:52:36Z",
}
これらの文字列をJava.util.Date objects
に変換するようにRetrofitまたはGsonに指示するにはどうすればよいですか?
Gson gson = new GsonBuilder()
.setDateFormat("yyyy-MM-dd'T'HH:mm:ss")
.create();
RestAdapter restAdapter = new RestAdapter.Builder()
.setEndpoint(API_BASE_URL)
.setConverter(new GsonConverter.create(gson))
.build();
または、Kotlinの同等物:
val gson = GsonBuilder().setDateFormat("yyyy-MM-dd'T'HH:mm:ss").create()
RestAdapter restAdapter = Retrofit.Builder()
.baseUrl(API_BASE_URL)
.addConverterFactory(GsonConverterFactory.create(gson))
.build()
.create(T::class.Java)
カスタマイズしたGsonパーサーを後付けに設定できます。詳細: Retrofit Website
Ondrejuの応答を見て、これをレトロフィット2に実装する方法を確認してください。
@gderacoの回答をレトロフィット2.0に更新:
Gson gson = new GsonBuilder()
.setDateFormat("yyyy-MM-dd'T'HH:mm:ss")
.create();
Retrofit retrofitAdapter = new Retrofit.Builder()
.baseUrl(API_BASE_URL)
.addConverterFactory(GsonConverterFactory.create(gson))
.build();
ここに私がそれをした方法があります:
Dateを拡張するDateTimeクラスを作成し、カスタムデシリアライザーを記述します。
public class DateTime extends Java.util.Date {
public DateTime(long readLong) {
super(readLong);
}
public DateTime(Date date) {
super(date.getTime());
}
}
次に、DateコンバーターとDateTimeコンバーターの両方を登録するデシリアライザーパーツについて説明します。
public static Gson gsonWithDate(){
final GsonBuilder builder = new GsonBuilder();
builder.registerTypeAdapter(Date.class, new JsonDeserializer<Date>() {
final DateFormat df = new SimpleDateFormat("yyyy-MM-dd");
@Override
public Date deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
try {
return df.parse(json.getAsString());
} catch (final Java.text.ParseException e) {
e.printStackTrace();
return null;
}
}
});
builder.registerTypeAdapter(DateTime.class, new JsonDeserializer<DateTime>() {
final DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
@Override
public DateTime deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
try {
return new DateTime(df.parse(json.getAsString()));
} catch (final Java.text.ParseException e) {
e.printStackTrace();
return null;
}
}
});
return builder.create();
}
RestAdapterを作成するときは、次のことを行います。
new RestAdapter.Builder().setConverter(gsonWithDate());
Fooは次のようになります。
class Foo {
Date date;
DateTime created_at;
}
Gsonは、カスタム形式での解析が不可能な場合、1つの日時形式(ビルダーで指定された形式)とiso8601のみを処理できます。そのため、解決策はカスタムデシリアライザーを作成することです。私が定義した問題を解決するには:
package stackoverflow.questions.q18473011;
import Java.util.Date;
public class Foo {
Date date;
Date created_at;
public Foo(Date date, Date created_at){
this.date = date;
this.created_at = created_at;
}
@Override
public String toString() {
return "Foo [date=" + date + ", created_at=" + created_at + "]";
}
}
このデシリアライザーで:
package stackoverflow.questions.q18473011;
import Java.lang.reflect.Type;
import Java.text.*;
import Java.util.Date;
import com.google.gson.*;
public class FooDeserializer implements JsonDeserializer<Foo> {
public Foo deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
String a = json.getAsJsonObject().get("date").getAsString();
String b = json.getAsJsonObject().get("created_at").getAsString();
SimpleDateFormat sdfDate = new SimpleDateFormat("yyyy-MM-dd");
SimpleDateFormat sdfDateWithTime = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss'Z'");
Date date, created;
try {
date = sdfDate.parse(a);
created = sdfDateWithTime.parse(b);
} catch (ParseException e) {
throw new RuntimeException(e);
}
return new Foo(date, created);
}
}
最後のステップは、適切なアダプターを使用してGson
インスタンスを作成することです。
package stackoverflow.questions.q18473011;
import com.google.gson.*;
public class Question {
/**
* @param args
*/
public static void main(String[] args) {
String s = "{ \"date\": \"2013-07-16\", \"created_at\": \"2013-07-16T22:52:36Z\"}";
GsonBuilder builder = new GsonBuilder();
builder.registerTypeAdapter(Foo.class, new FooDeserializer());
Gson gson = builder.create();
Foo myObject = gson.fromJson(s, Foo.class);
System.out.println("Result: "+myObject);
}
}
私の結果:
Result: Foo [date=Tue Jul 16 00:00:00 CEST 2013, created_at=Tue Jul 16 22:52:36 CEST 2013]
文字通り、作成しているクラスに「created_at」という名前のDateオブジェクトがすでにある場合、これは簡単です。
Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd'T'HH:mm:ss'Z'").create();
YourObject parsedObject1 = gson.fromJson(JsonStringYouGotSomehow, YourObject.class);
これで完了です。複雑なオーバーライドは必要ありません。
次のように2つの新しいクラスを定義できます。
import Java.util.Date;
public class MyDate extends Date {
}
そして
import Java.util.Date;
public class CreatedAtDate extends Date {
}
POJOは次のようになります。
import MyDate;
import CreatedAtDate;
public class Foo {
MyDate date;
CreatedAtDate created_at;
}
最後に、カスタムデシリアライザーを設定します。
public class MyDateDeserializer implements JsonDeserializer<Date> {
public static final SimpleDateFormat sServerDateDateFormat = new SimpleDateFormat("yyyy-MM-dd");
@Override
public MyDate deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
if (json != null) {
final String jsonString = json.getAsString();
try {
return (MyDate) sServerDateDateFormat.parse(jsonString);
} catch (ParseException e) {
e.printStackTrace();
}
}
return null;
}
}
そして
GsonBuilder builder = new GsonBuilder();
builder.registerTypeAdapter(MyDate.class, new MyDateDeserializer());