Jacksonに単純なパターン(「dd-MM-yyyy」など)に従ってJoda DateTimeオブジェクトをシリアル化するにはどうすればよいですか?
私はもう試した:
@JsonSerialize(using=DateTimeSerializer.class)
private final DateTime date;
私も試しました:
ObjectMapper mapper = new ObjectMapper()
.getSerializationConfig()
.setDateFormat(df);
ありがとう!
これは、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/
マッピングしているオブジェクトで:
@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));
}
}
@Kimbleが言ったように、Jackson 2では、デフォルトのフォーマットを使用するのは非常に簡単です。 JodaModule
にObjectMapper
を登録するだけです。
ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(new JodaModule());
DateTime
のカスタムシリアライゼーション/デシリアライゼーションの場合、独自のStdScalarSerializer
およびStdScalarDeserializer
;を実装する必要があります。かなり複雑ですが、とにかく。
たとえば、UTCタイムゾーンでDateTime
を使用するISODateFormat
シリアライザーは次のとおりです。
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());
https://stackoverflow.com/a/10835114/111351
各日付フィールドに注釈を付けることができますが、オブジェクトマッパーのグローバル構成を行う方が適切です。 jacksonを使用する場合、次のようにスプリングを設定できます。
<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は必要なフォーマットを使用できます。
私は同様の問題に遭遇し、私の解決策は上記よりもはるかに明確です。
私は単純にパターンを使用しました@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を使用します
一方、JodaModuleがクラスパスにある場合、JacksonはJodaモジュールを自動的に登録します。 jackson-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"}
Spring Bootを使用する場合は、モジュールをコンテキストに追加する必要があり、このように構成に追加されます。
@Bean
public Module jodaTimeModule() {
return new JodaModule();
}
また、新しいJava8タイムモジュールjsr-310を使用する場合。
@Bean
public Module jodaTimeModule() {
return new JavaTimeModule();
}
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);
}
}
このクラスは、Joda DateTimeのtoString()メソッドを使用してデータをシリアル化します。
Rusty Kuntzが提案したアプローチは私の場合に最適です。
私はJava 8を使用していますが、これは私のために働いた:
pom.xmlに依存関係を追加します<dependency> <groupId>com.fasterxml.jackson.datatype</groupId> <artifactId>jackson-datatype-jsr310</artifactId> <version>2.4.0</version> </dependency>
objectMapperにJodaModuleを追加しますObjectMapper mapper = new ObjectMapper(); mapper.registerModule(new JodaModule());