GSON - формат даты

187

Я пытаюсь иметь пользовательский формат даты в выводе 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()не работает :(

Стефан Пиетт
источник
Вы проверили, что вы импортируете правильный Date-класс во всех местах? Также пример кода будет полезен.
ML
Что вы подразумеваете под не работает? Что ты на самом деле видишь?
АликЭльзин-килака

Ответы:

306

Похоже, вам нужно определить форматы для части даты и времени или использовать форматирование на основе строк. Например:

Gson gson = new GsonBuilder()
   .setDateFormat("EEE, dd MMM yyyy HH:mm:ss zzz").create();

или используя java.text.DateFormat

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());
ML
источник
Хорошо, это работает лучше с вашим способом. Но как я могу отображать метки времени с этим?
Стефан Пиетт
6
Я использую Gson 2.2.4 и Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd").create();работает.
Рамшар
с использованием Gson 2.2.2 и Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd").create();работает
user2601995
66
Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSSZ").create();

Выше формат кажется мне лучше, поскольку он имеет точность до миллис.

TJ-Выемка
источник
1
@Abhi - Почему нужно цитировать 'Z'?
TJ-Выемка
1
Я попробовал без кавычек и гсон выкинул исключение, что дата не разбирается. У вас работает без кавычек?
Абхи
7
Это НЕ правильное решение ... Z означает часовой пояс и представляет часовой пояс UTC, если вы жестко закодируете его таким образом, вы потеряете эту информацию
Billda
5

Как указывал ML, 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, хотя.

Milanka
источник
4

В случае, если вы ненавидите внутренние классы, используя функциональный интерфейс, вы можете написать меньше кода в Java 8 с лямбда-выражением.

JsonDeserializer<Date> dateJsonDeserializer = 
     (json, typeOfT, context) -> json == null ? null : new Date(json.getAsLong());
Gson gson = new GsonBuilder().registerTypeAdapter(Date.class,dateJsonDeserializer).create();
Картик
источник
Кажется, что нулевая проверка может быть опущена (Gson делает это сам):Gson gson = new GsonBuilder().registerTypeAdapter(Date.class, (JsonDeserializer) (json, typeOfT, context) -> new Date(json.getAsLong())).create();
gmk57
0

Это не будет работать вообще. В JSON нет типа даты. Я бы порекомендовал сериализовать в ISO8601 и обратно (для агностики форматов и JS-совместимости). Учтите, что вы должны знать, какие поля содержат даты.

Michael-O
источник
0

Вы можете указать свой формат 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();
Абхиджит чакра
источник
0

Это ошибка . В настоящее время вы должны либо установить a, timeStyleлибо использовать один из вариантов, описанных в других ответах.

Marcono1234
источник
0

Я на Gson 2.8.6 и обнаружил эту ошибку сегодня.

Мой подход позволяет всем нашим существующим клиентам (мобильным / веб / и т. Д.) Продолжать функционировать в том виде, в каком они были, но добавляет некоторую обработку для тех, кто использует 24-часовые форматы, и также допускает использование миллисера.

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, но это должно быть довольно редким случаем.

Джо Сиарамитаро
источник