Как сериализовать Joda DateTime с процессором Jackson JSON?
Как я могу заставить Jackson сериализовать объект Joda DateTime в соответствии с простым шаблоном (например, "dd-MM-yyyy" )?
Я пробовал:
@JsonSerialize(using=DateTimeSerializer.class)
private final DateTime date;
Я также пробовал:
ObjectMapper mapper = new ObjectMapper()
.getSerializationConfig()
.setDateFormat(df);
Спасибо!
Ответы
Ответ 1
Это стало очень легко с 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/
Ответ 2
В отображаемом объекте:
@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));
}
}
Ответ 3
Как сказал @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());
Ответ 4
Простое решение
Я столкнулся с подобной проблемой, и мое решение намного яснее, чем выше.
Я просто использовал шаблон в аннотации @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);
}
Мы используем Джексон 2.5.4
Ответ 5
fooobar.com/questions/56229/...
Хотя вы можете поместить аннотацию для каждого поля даты, лучше сделать глобальную конфигурацию для вашего объекта mapper. Если вы используете jackson, вы можете настроить spring следующим образом:
<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 может использовать любой формат, который вам нужен.
Ответ 6
Тем временем Джексон автоматически регистрирует модуль Joda, когда JodaModule находится в classpath. Я просто добавил джексон-datatype-joda в Maven, и это сработало мгновенно.
<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"}
Ответ 7
Похоже, что для 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.
Подход, предложенный Расти Кунцем, отлично подходит для моего дела.
Ответ 8
Для тех, у кого есть Spring Boot, вам нужно добавить модуль в свой контекст, и он будет добавлен в вашу конфигурацию следующим образом.
@Bean
public Module jodaTimeModule() {
return new JodaModule();
}
И если вы хотите использовать новый java8-модуль времени jsr-310.
@Bean
public Module jodaTimeModule() {
return new JavaTimeModule();
}
Ответ 9
Я использую 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());