Как сериализовать Joda DateTime с процессором Jackson JSON?


118

Как мне заставить Джексона сериализовать мой объект Joda DateTime в соответствии с простым шаблоном (например, «дд-мм-гггг»)?

Я пробовал:

@JsonSerialize(using=DateTimeSerializer.class)
private final DateTime date;

Я также пробовал:

ObjectMapper mapper = new ObjectMapper()
    .getSerializationConfig()
    .setDateFormat(df);

Спасибо!


Оба вышеперечисленного должны также работать (@JsonSerialize должно подразумевать, что это поле должно быть сериализовано; и формат даты также в идеале должен применяться к Joda), поэтому вы можете зарегистрировать ошибку Jira на jira.codehaus.org/browse/JACKSON .
StaxMan 02

Я понимаю, что этот вопрос задан некоторое время назад, но для справки в будущем objectMapper.getSerializationConfig (). SetDateFormat (df) теперь устарел. ObjectMapper.setDateFormat (df) теперь предлагается.
Патрик

Ответы:


146

Это стало очень просто с Jackson 2.0 и модулем Joda.

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());

Зависимость от Maven:

<dependency>
  <groupId>com.fasterxml.jackson.datatype</groupId>
  <artifactId>jackson-datatype-joda</artifactId>
  <version>2.1.1</version>
</dependency>  

Код и документация: https://github.com/FasterXML/jackson-datatype-joda

Двоичные файлы: http://repo1.maven.org/maven2/com/fasterxml/jackson/datatype/jackson-datatype-joda/


13
Не могли бы вы подробнее рассказать об этом? Куда идет этот код и как используется экземпляр ObjectMapper?
Пол

Вам следует задать конкретные вопросы относительно использования API Джексона и Maven. То, куда идет код, зависит от того, используете ли вы какие-либо фреймворки / как вы загружаете свое приложение.
Kimble

4
когда я добавляю, что получаю ошибку компиляции «несовместимые типы: JodaModule не может быть преобразован в модуль» - метод ожидает org.codehaus.jackson.map.Module, но JodaModule не имеет этого в своей иерархии, так как же это могло работать?
Martin Charlesworth

4
Многообещающе, но в настоящее время почти бесполезно, поскольку мы не можем указать формат даты для синтаксического анализа :(
Винсент Мимун-Прат

10
Вам необходимо отключить сериализацию как отметки времени objectMapper.disable (SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
MarekM 02

74

В отображаемом объекте:

@JsonSerialize(using = CustomDateSerializer.class)
public DateTime getDate() { ... }

В CustomDateSerializer:

public class CustomDateSerializer extends JsonSerializer<DateTime> {

    private static DateTimeFormatter formatter = 
        DateTimeFormat.forPattern("dd-MM-yyyy");

    @Override
    public void serialize(DateTime value, JsonGenerator gen, 
                          SerializerProvider arg2)
        throws IOException, JsonProcessingException {

        gen.writeString(formatter.print(value));
    }
}

4
Как зарегистрировать CustomDateSerializer в Spring MVC 3?
digz6666

1
Я не знаю о Spring MVC3, но вы можете добавить собственный сериализатор к объекту сопоставления Джексона. См. Эту запись в Джексоне. Вам необходимо создать простой модуль для этого SimpleModule isoDateTimeModule = new SimpleModule ("ISODateTimeModule", new Version (1, 0, 0, null)); isoDateTimeModule.addSerializer (новый JsonISODateTimeFormatSerializer ()); mapper.registerModule (isoDateTimeModule);
Гийом Белроз

1
@ digz6666 см. мой ответ для Spring MVC 3: stackoverflow.com/questions/7854030/…
Арам Кочарян,

Примечание. Убедитесь, что вы используете вышеуказанные классы from, org.codehaus.jacksonа не com.fasterxml.jackson.core. Использование второго пакета у меня не сработало. Фактически, мое приложение в Джерси даже не соблюдало @JsonSerializedаннотацию.
Кевин Мередит

Ответ сработал для меня. Но для этого потребуется эта аннотация во всех полях. Есть ли какая-то глобальная конфигурация для этого, которая работает с Джексоном?
Taher

26

Как сказал @Kimble, с Jackson 2 очень легко использовать форматирование по умолчанию; просто зарегистрируйтесь JodaModuleна своем ObjectMapper.

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());

Для пользовательской сериализации / десериализации DateTimeвам необходимо реализовать свои собственные StdScalarSerializerиStdScalarDeserializer ; это довольно запутанно, но все равно.

Например, вот DateTimeсериализатор, который использует ISODateFormatчасовой пояс UTC:

public class DateTimeSerializer extends StdScalarSerializer<DateTime> {

    public DateTimeSerializer() {
        super(DateTime.class);
    }

    @Override
    public void serialize(DateTime dateTime,
                          JsonGenerator jsonGenerator,
                          SerializerProvider provider) throws IOException, JsonGenerationException {
        String dateTimeAsString = ISODateTimeFormat.withZoneUTC().print(dateTime);
        jsonGenerator.writeString(dateTimeAsString);
    }
}

И соответствующий десериализатор:

public class DateTimeDesrializer extends StdScalarDeserializer<DateTime> {

    public DateTimeDesrializer() {
        super(DateTime.class);
    }

    @Override
    public DateTime deserialize(JsonParser jsonParser,
                                DeserializationContext deserializationContext) throws IOException, JsonProcessingException {
        try {
            JsonToken currentToken = jsonParser.getCurrentToken();
            if (currentToken == JsonToken.VALUE_STRING) {
                String dateTimeAsString = jsonParser.getText().trim();
                return ISODateTimeFormat.withZoneUTC().parseDateTime(dateTimeAsString);
            }
        } finally {
            throw deserializationContext.mappingException(getValueClass());
        }
    }

Затем свяжите их вместе модулем:

public class DateTimeModule extends SimpleModule {

    public DateTimeModule() {
        super();
        addSerializer(DateTime.class, new DateTimeSerializer());
        addDeserializer(DateTime.class, new DateTimeDeserializer());
    }
}

Затем зарегистрируйте модуль на своем ObjectMapper:

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new DateTimeModule());

объект deserilazed DatatIme находится в часовом поясе UTC или часовом поясе браузеров, пожалуйста, помогите мне
user3354457

Регистрирует ли это картограф (и модули) глобально?
raffian

16

Простое решение

Я столкнулся с подобной проблемой, и мое решение намного яснее, чем указано выше.

Я просто использовал узор в @JsonFormat аннотации

В основном у моего класса есть DateTimeполе, поэтому я помещаю аннотацию вокруг получателя:

@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
public DateTime getDate() {
    return date;
}

Я сериализую класс с помощью ObjectMapper

    ObjectMapper mapper = new ObjectMapper();
    mapper.registerModule(new JodaModule());
    mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
    ObjectWriter ow = mapper.writer();
    try {
        String logStr = ow.writeValueAsString(log);
        outLogger.info(logStr);
    } catch (IOException e) {
        logger.warn("JSON mapping exception", e);
    }

Мы используем Jackson 2.5.4


15

https://stackoverflow.com/a/10835114/1113510

Хотя вы можете разместить аннотацию для каждого поля даты, лучше сделать глобальную конфигурацию для вашего сопоставителя объектов. Если вы используете Джексон, вы можете настроить свою пружину следующим образом:

<bean id="jacksonObjectMapper" class="com.company.CustomObjectMapper" />

<bean id="jacksonSerializationConfig" class="org.codehaus.jackson.map.SerializationConfig"
    factory-bean="jacksonObjectMapper" factory-method="getSerializationConfig" >
</bean>

Для CustomObjectMapper:

public class CustomObjectMapper extends ObjectMapper {

    public CustomObjectMapper() {
        super();
        configure(Feature.WRITE_DATES_AS_TIMESTAMPS, false);
        setDateFormat(new SimpleDateFormat("EEE MMM dd yyyy HH:mm:ss 'GMT'ZZZ (z)"));
    }
}

Конечно, SimpleDateFormat может использовать любой нужный вам формат.


4
setDateFormat(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssz"));сделал трюк для меня,
спасибо

8

Тем временем Джексон автоматически регистрирует модуль Joda, когда JodaModule находится в пути к классам. Я просто добавил в Maven jackson-datatype-joda, и он сразу заработал.

<dependency>
  <groupId>com.fasterxml.jackson.datatype</groupId>
  <artifactId>jackson-datatype-joda</artifactId>
  <version>2.8.7</version>
</dependency>

Вывод JSON:

{"created" : "2017-03-28T05:59:27.258Z"}

6

Для тех, у кого есть Spring Boot, вам нужно добавить модуль в свой контекст, и он будет добавлен в вашу конфигурацию следующим образом.

@Bean
public Module jodaTimeModule() {
    return new JodaModule();
}

И если вы хотите использовать новый модуль времени java8 jsr-310.

@Bean
public Module jodaTimeModule() {
    return new JavaTimeModule();
}

3

Похоже, что для Jackson 1.9.12 такой возможности по умолчанию нет из-за:

public final static class DateTimeSerializer
    extends JodaSerializer<DateTime>
{
    public DateTimeSerializer() { super(DateTime.class); }

    @Override
    public void serialize(DateTime value, JsonGenerator jgen, SerializerProvider provider)
        throws IOException, JsonGenerationException
    {
        if (provider.isEnabled(SerializationConfig.Feature.WRITE_DATES_AS_TIMESTAMPS)) {
            jgen.writeNumber(value.getMillis());
        } else {
            jgen.writeString(value.toString());
        }
    }

    @Override
    public JsonNode getSchema(SerializerProvider provider, java.lang.reflect.Type typeHint)
    {
        return createSchemaNode(provider.isEnabled(SerializationConfig.Feature.WRITE_DATES_AS_TIMESTAMPS)
                ? "number" : "string", true);
    }
}

Этот класс сериализует данные с помощью метода toString () Joda DateTime.

Подход, предложенный Расти Кунцем, идеально подходит для моего случая.


Вы по-прежнему можете зарегистрировать пользовательский сериализатор и десериализатор, и они будут иметь приоритет над реализацией по умолчанию.
StaxMan

2

Я использую Java 8, и у меня это сработало.

Добавьте зависимость от pom.xml

<dependency>
    <groupId>com.fasterxml.jackson.datatype</groupId>
    <artifactId>jackson-datatype-jsr310</artifactId>
    <version>2.4.0</version>
</dependency>

и добавьте JodaModule на свой ObjectMapper

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.