使用Jackson JSR310模块反序列化LocalDateTime

时间:2015-04-10 22:24:17

标签: java jackson java-time jsr310

我正在使用the Jackson Datatype JSR310 page描述的库,但我仍然难以让它工作。

我已配置以下bean:

@Bean
@Primary
public ObjectMapper objectMapper() {
    ObjectMapper mapper = new ObjectMapper();
    mapper.registerModule(new JSR310Module());
    return mapper;
}

当我调用我的REST API时,日期格式输出为yyyy-MM-dd'T'HH:ss.SSSSSS,例如2015-04-11T00:10:38.905847。这可以通过我的AngularJS代码处理得很好。

当我想向REST API提交内容时,日期会发布为yyyy-MM-dd'T'HH:mm:ss.SSS'Z',例如2015-04-09T08:30:00.000Z

杰克逊最后一直抱怨'Z'。如果我查看文档中的LocalDateTimeDeserializer,它会使用DateTimeFormatter.ISO_LOCAL_DATE_TIME沸腾到ISO_LOCAL_DATE'T'ISO_LOCAL_TIME,并提到它没有覆盖区域。

所以我想我应该在我正在创建的DateFormat上设置ObjectMapper

@Bean
@Primary
public ObjectMapper objectMapper() {
    ObjectMapper mapper = new ObjectMapper();
    mapper.registerModule(new JSR310Module());
    mapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS'Z'"));
    return mapper;
}

但这没有任何作用。我将其更改为像yyyy-MM-dd这样的简单内容,但序列化日期保留为先前格式,反序列化也不受影响。

我在这里做错了什么才能让它发挥作用?据我所知,我的JavaScript代码中的日期格式是ISO 8601格式......

3 个答案:

答案 0 :(得分:23)

没有必要编写自己的序列化程序。它足以使用默认的一个,但是使用另一种格式(time_zone一个)创建一个实例,以便超出部分只是剪切:

new LocalDateTimeDeserializer(DateTimeFormatter.ISO_DATE_TIME)

在我的情况下,我有一个contextResolver这样的配置来实现:

@Service 
@Provider
public class ObjectMapperContextResolver implements ContextResolver<ObjectMapper> {  
    private final ObjectMapper mapper;

    public ObjectMapperContextResolver() {
        mapper = new ObjectMapper();
        JavaTimeModule javaTimeModule=new JavaTimeModule();
        // Hack time module to allow 'Z' at the end of string (i.e. javascript json's) 
        javaTimeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ISO_DATE_TIME));
        mapper.registerModule(javaTimeModule);
        mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
    }

    @Override
    public ObjectMapper getContext(Class<?> type) {
        return mapper;
    }  
}

答案 1 :(得分:5)

现在LocalDateTimeDeserializer似乎不尊重为对象映射器设置的日期格式。

要使其有效,您可以覆盖LocalDateTimeDeserializer或切换为使用ZoneDateTime来处理&#39; Z&#39; char最后。

以下是一个例子:

public class Java8DateFormat {
    public static void main(String[] args) throws IOException {
        final ObjectMapper mapper = new ObjectMapper();
        mapper.registerModule(new JSR310Module());
        // mapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss.SSS'Z'"));

        final String date = mapper.writeValueAsString(new Date());
        System.out.println(date);
        System.out.println(mapper.readValue(date, ZonedDateTime.class));
    }
}

输出:

"2015-04-11T18:24:47.815Z"
2015-04-11T18:24:47.815Z[GMT]

答案 2 :(得分:1)

Hibernate 4,Spring 4 - REST WS,Client - Spring Boot 1.5.2。在我的例子中,我在Entity ZonedDateTime 类中用于在数据库中映射 Timestamp 。 Hibernate以及Spring Boot REST工作正常。我只能将库添加到 pom 文件中:

    <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-annotations -->
    <dependency>
        <groupId>com.fasterxml.jackson.core</groupId>
        <artifactId>jackson-annotations</artifactId>
        <version>${jackson.version}</version>
    </dependency>

    <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.datatype/jackson-datatype-jsr310 -->
    <dependency>
        <groupId>com.fasterxml.jackson.datatype</groupId>
        <artifactId>jackson-datatype-jsr310</artifactId>
        <version>${jackson.version}</version>
    </dependency>

所以我想,该转换器也是在Spring内部为 LocalDateTime 实现的。 jackson.version 是最新的。