17

我正在使用 Jackson 2.8,并且需要与不允许在 ISO 8601 时间戳内毫秒数的 API 进行通信。

预期的格式是这样的:"2016-12-24T00:00:00Z"

我正在使用 Jackson 的 JavaTimeModule 并WRITE_DATES_AS_TIMESTAMPS设置为false.

但这将打印毫秒。

所以我尝试使用objectMapper.setDateFormatwhich 并没有改变任何东西。

我目前的解决方法是这样的:

ObjectMapper om = new ObjectMapper();

DateTimeFormatter dtf = new DateTimeFormatterBuilder()
    .appendInstant(0)
    .toFormatter();

JavaTimeModule jtm = new JavaTimeModule();
jtm.addSerializer(Instant.class, new JsonSerializer<Instant>() {
    @Override
    public void serialize(Instant value, JsonGenerator gen, SerializerProvider serializers) throws IOException, JsonProcessingException {
        gen.writeString(dtf.format(value));
    }
});

om.registerModule(jtm);

我正在覆盖适用的默认序列化程序Instant.class


有没有什么好方法使用一些配置参数来解决这个问题?

4

3 回答 3

10

更新:

只需在属性@JsonFormat上方添加日期格式的注释即可。Instant这很容易。

如果你有一个 ObjectMapperJavaTimeModule类似下一个:

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

如果您有一个具有Instant属性的类,则应添加@JsonFormat注释并放置没有毫秒的日期模式。就像下一个:

public static class TestDate {

    @JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd hh:mm:ss", timezone = "UTC")
    Instant instant;

    //getters & setters
}

因此,如果您将一个对象序列化为 Json,它将完美地工作:

String json = mapper.writeValueAsString(testDate);
System.out.println(json); 

输出

{“即时”:“2016-11-10 06:03:06”}


旧答案。我不知道为什么,但它不能正常工作:

您可以使用Jackson2ObjectMapperBuilder来构建它。

您只需要添加所需的 dateFormat 即可。这将是下一个:

DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");

ObjectMapper mapper = Jackson2ObjectMapperBuilder
            .json()       
            .featuresToDisable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS) 
            .modules(new JavaTimeModule())
            .dateFormat(dateFormat)
            .build();
于 2016-11-08T13:09:01.117 回答
6

这是您可以全局设置的替代方法,但需要您使用ZonedDateTime即时格式化程序,因为我们无法为InstantJava 时间模块提供的序列化程序设置格式。

您不会看到即时使用分区日期时间的任何副作用,因为杰克逊单独序列化区域 ID 并默认禁用。所以从技术上讲,这类似于将格式化程序应用于Instant.

当以这种方式使用时,ZonedDateTime序列化程序将序列化委托给InstantBaseSerializer并使用指定的自定义格式。

@RunWith(JUnit4.class)
public class InstantNoMillisTest {

    private ObjectMapper objectMapper;

    @Before
    public void init() {
        JavaTimeModule module = new JavaTimeModule();
        ZonedDateTimeSerializer zonedDateTimeSerializer = new ZonedDateTimeSerializer(new DateTimeFormatterBuilder().appendInstant(0).toFormatter());
        module.addSerializer(ZonedDateTime.class, zonedDateTimeSerializer);
        module.addDeserializer(ZonedDateTime.class, InstantDeserializer.ZONED_DATE_TIME);

        objectMapper = Jackson2ObjectMapperBuilder.json()
                .modules(module)
                .featuresToDisable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
                .build();
    }

    @Test
    public void serialize() throws IOException {
        ZonedDateTime zonedDateTime = ZonedDateTime.now();
        String noMillis = objectMapper.writeValueAsString(zonedDateTime);
        System.out.print(noMillis);
    }

    @Test
    public void deserialize() throws IOException {
        String dateTime = "\"2017-10-26T12:54:59Z\"";
        ZonedDateTime noMillis = objectMapper.readValue(dateTime, ZonedDateTime.class);
        System.out.print(noMillis);
    }
}
于 2016-11-11T14:04:10.710 回答
5

这是格式化字段的一些 Kotlin 代码Instant,因此它们不会包含毫秒,您可以使用自定义日期格式化程序

ObjectMapper().apply {
        val javaTimeModule = JavaTimeModule()
        javaTimeModule.addSerializer(Instant::class.java, Iso8601WithoutMillisInstantSerializer())
        registerModule(javaTimeModule)
        disable(WRITE_DATES_AS_TIMESTAMPS)
    }

private class Iso8601WithoutMillisInstantSerializer
        : InstantSerializer(InstantSerializer.INSTANCE, false, DateTimeFormatterBuilder().appendInstant(0).toFormatter())
于 2019-01-03T15:09:55.477 回答