我有一个Android应用程序,它接收来自Web服务的Json响应。其中一个响应是一个内部有日期的json字符串。我以#34; 1476399300000"等数字的形式得到日期。当我尝试使用GSON
创建一个对象时,我收到此错误:
无法解析日期[" 1476399300000']:无效的时区指示符' 0' (在偏移0处)
双方正在使用java.util.Date
如何解决此问题?
答案 0 :(得分:16)
值1476399300000
从Unix epoch开头看起来像ms。只需在Gson中添加一个类型适配器:
final class UnixEpochDateTypeAdapter
extends TypeAdapter<Date> {
private static final TypeAdapter<Date> unixEpochDateTypeAdapter = new UnixEpochDateTypeAdapter();
private UnixEpochDateTypeAdapter() {
}
static TypeAdapter<Date> getUnixEpochDateTypeAdapter() {
return unixEpochDateTypeAdapter;
}
@Override
public Date read(final JsonReader in)
throws IOException {
// this is where the conversion is performed
return new Date(in.nextLong());
}
@Override
@SuppressWarnings("resource")
public void write(final JsonWriter out, final Date value)
throws IOException {
// write back if necessary or throw UnsupportedOperationException
out.value(value.getTime());
}
}
配置您的Gson实例:
final Gson gson = new GsonBuilder()
.registerTypeAdapter(Date.class, getUnixEpochDateTypeAdapter())
.create();
Gson
个实例是线程安全的,UnixEpochDateTypeAdapter
是,并且可以作为一个全局实例存在。例如:
final class Mapping {
final Date date = null;
}
final String json = "{\"date\":1476399300000}";
final Mapping mapping = gson.fromJson(json, Mapping.class);
System.out.println(mapping.date);
System.out.println(gson.toJson(mapping));
会给出以下输出:
Fri Oct 14 01:55:00 EEST 2016
{“date”:1476399300000}
请注意,类型适配器配置为覆盖默认的Gson日期类型适配器。因此,您可能需要使用更复杂的分析来检测是否只是Unix时代的ms。另请注意,您可以使用JsonDeserializer
,但后者以JSON树方式工作,而类型适配器以流式方式工作,这种方式更有效,可能不会累积中间结果。
编辑:
此外,它可能看起来令人困惑,但Gson可以为基元进行值转换。尽管您的有效负载具有字符串值,但JsonReader.nextLong()
可以将字符串原语读取为长值。因此,UnixEpochDateTypeAdapter.write
应为out.value(String.valueOf(value.getTime()));
,以便不修改JSON文字。
还有一个更短的解决方案(使用JSON内存中的树而不是数据流),这很简单:
final Gson builder = new GsonBuilder()
.registerTypeAdapter(Date.class, new JsonDeserializer<Date>() {
public Date deserialize(JsonElement jsonElement, Type type, JsonDeserializationContext context) throws JsonParseException {
return new Date(jsonElement.getAsJsonPrimitive().getAsLong());
}
})
.create();