我正在尝试在 Gson 输出中使用自定义日期格式,但
.setDateFormat(DateFormat.FULL)
似乎不起作用,它与 .registerTypeAdapter(Date.class, new DateSerializer())
相同。
就好像Gson不关心“Date”这个对象,按照它的方式打印出来
我怎样才能改变它?
谢谢
编辑:
@Entity
public class AdviceSheet {
public Date lastModif;
[...]
}
public void method {
Gson gson = new GsonBuilder().setDateFormat(DateFormat.LONG).create();
System.out.println(gson.toJson(adviceSheet);
}
我一直用
java.util.Date
; setDateFormat()
不起作用:(
看来您需要为日期和时间部分定义格式或使用基于字符串的格式。例如:
Gson gson = new GsonBuilder()
.setDateFormat("EEE, dd MMM yyyy HH:mm:ss zzz").create();
Gson gson = new GsonBuilder()
.setDateFormat(DateFormat.FULL, DateFormat.FULL).create();
或者用序列化器来做:
我相信格式化程序不能生成时间戳,但是这个序列化器/反序列化器对似乎可以工作
JsonSerializer<Date> ser = new JsonSerializer<Date>() {
@Override
public JsonElement serialize(Date src, Type typeOfSrc, JsonSerializationContext
context) {
return src == null ? null : new JsonPrimitive(src.getTime());
}
};
JsonDeserializer<Date> deser = 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, ser)
.registerTypeAdapter(Date.class, deser).create();
如果使用 Java 8 或更高版本,您应该像这样使用上面的序列化器/反序列化器:
JsonSerializer<Date> ser = (src, typeOfSrc, context) -> src == null ? null
: new JsonPrimitive(src.getTime());
JsonDeserializer<Date> deser = (jSon, typeOfT, context) -> jSon == null ? null : new Date(jSon.getAsLong());
Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSZ").create();
以上格式对我来说似乎更好,因为它的精度高达毫秒。
作为 M.L.指出,JsonSerializer 在这里工作。但是,如果您正在格式化数据库实体,请使用 java.sql.Date 来注册您的序列化程序。不需要解串器。
Gson gson = new GsonBuilder()
.registerTypeAdapter(java.sql.Date.class, ser).create();
此错误报告可能相关:http://code.google.com/p/google-gson/issues/detail?id=230。不过我使用的是 1.7.2 版本。
如果你讨厌内部类,通过利用功能接口,你可以在 Java 8 中使用 lambda 表达式编写更少的代码。
JsonDeserializer<Date> dateJsonDeserializer =
(json, typeOfT, context) -> json == null ? null : new Date(json.getAsLong());
Gson gson = new GsonBuilder().registerTypeAdapter(Date.class,dateJsonDeserializer).create();
您可以在此方法中指定您的格式
Gson gson = builder.setDateFormat("yyyy-MM-dd").create();
而不是yyyy-MM-dd
您可以使用任何其他格式
GsonBuilder builder = new GsonBuilder();
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.setDateFormat("yyyy-MM-dd").create();
这是一个bug。目前,您要么也必须设置
timeStyle
,要么使用其他答案中描述的替代方案之一。
我在 Gson 2.8.6 上,今天发现了这个错误。
我的方法允许我们所有现有的客户端(移动/网络/等)继续像以前一样运行,但为那些使用 24 小时格式的客户端添加一些处理,并且也允许 millis,这是一个很好的衡量标准。
Gson rawGson = new Gson();
SimpleDateFormat fmt = new SimpleDateFormat("MMM d, yyyy HH:mm:ss")
private class DateDeserializer implements JsonDeserializer<Date> {
@Override
public Date deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context)
throws JsonParseException {
try {
return new rawGson.fromJson(json, Date.class);
} catch (JsonSyntaxException e) {}
String timeString = json.getAsString();
log.warning("Standard date deserialization didn't work:" + timeString);
try {
return fmt.parse(timeString);
} catch (ParseException e) {}
log.warning("Parsing as json 24 didn't work:" + timeString);
return new Date(json.getAsLong());
}
}
Gson gson = new GsonBuilder()
.registerTypeAdapter(Date.class, new DateDeserializer())
.create();
我保持序列化相同,因为所有客户都理解标准的 json 日期格式。
通常,我不认为使用 try/catch 块来管理流控制是一种好的做法,但这应该是一种相当罕见的情况。
这根本行不通。 JSON 中没有日期类型。我建议来回序列化为 ISO 8601(用于格式不可知论和 JS 兼容)。考虑到您必须知道哪些字段包含日期。