Spring Boot和Feign中使用Java 8時間日期API(LocalDate等)的序列化問題

LocalDateLocalTimeLocalDateTime是Java 8開始提供的時間日期API,主要用來優化Java 8之前對於時間日期的處理操做。然而,咱們在使用Spring Boot或使用Spring Cloud Feign的時候,每每會發現使用請求參數或返回結果中有 LocalDateLocalTimeLocalDateTime的時候會發生各類問題。本文咱們就來講說這種狀況下出現的問題,以及如何解決。

問題現象

先來看看症狀。好比下面的例子:java

@SpringBootApplication
public class Application {

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }

    @RestController
    class HelloController {

        @PostMapping("/user")
        public UserDto user(@RequestBody UserDto userDto) throws Exception {
            return userDto;
        }

    }

    @Data
    @NoArgsConstructor
    @AllArgsConstructor
    static class UserDto {

        private String userName;
        private LocalDate birthday;

    }

}

上面的代碼構建了一個簡單的Spring Boot Web應用,它提供了一個提交用戶信息的接口,用戶信息中包含了LocalDate類型的數據。此時,若是咱們使用Feign來調用這個接口的時候,會獲得以下錯誤:git

2018-03-13 09:22:58,445 WARN  [http-nio-9988-exec-3] org.springframework.web.servlet.mvc.support.DefaultHandlerExceptionResolver - Failed to read HTTP message: org.springframework.http.converter.HttpMessageNotReadableException: JSON parse error: Can not construct instance of java.time.LocalDate: no suitable constructor found, can not deserialize from Object value (missing default constructor or creator, or perhaps need to add/enable type information?); nested exception is com.fasterxml.jackson.databind.JsonMappingException: Can not construct instance of java.time.LocalDate: no suitable constructor found, can not deserialize from Object value (missing default constructor or creator, or perhaps need to add/enable type information?)
 at [Source: java.io.PushbackInputStream@67064c65; line: 1, column: 63] (through reference chain: java.util.ArrayList[0]->com.didispace.UserDto["birthday"])

分析解決

對於上面的錯誤信息JSON parse error: Can not construct instance of java.time.LocalDate: no suitable constructor found, can not deserialize from Object value,熟悉Spring MVC的童鞋應該立刻就能定位錯誤與LocalDate的反序列化有關。可是,依然會有不少讀者會被這段錯誤信息java.util.ArrayList[0]->com.didispace.UserDto["birthday"]所困惑。咱們命名提交的UserDto["birthday"]是個LocalDate對象嘛,跟ArrayList列表對象有啥關係呢?github

咱們不妨經過postman等手工發一個請求看看服務端返回的是什麼?好比你能夠按下圖發起一個請求:web

從上圖中咱們就能夠理解上面我所提到的困惑了,實際上默認狀況下Spring MVC對於LocalDate序列化成了一個數組類型,而Feign在調用的時候,仍是按照ArrayList來處理,因此天然沒法反序列化爲LocalDate對象了。spring

解決方法數組

爲了解決上面的問題很是簡單,由於jackson也爲此提供了一整套的序列化方案,咱們只須要在pom.xml中引入jackson-datatype-jsr310依賴,具體以下:bash

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

注意:在設置了spring boot的parent的狀況下不須要指定具體的版本,也不建議指定某個具體版本mvc

在該模塊中封裝對Java 8的時間日期API序列化的實現,其具體實如今這個類中:com.fasterxml.jackson.datatype.jsr310.JavaTimeModule(注意:一些較早版本瘋轉在這個類中「com.fasterxml.jackson.datatype.jsr310.JSR310Module)。在配置了依賴以後,咱們只須要在上面的應用主類中增長這個序列化模塊,同時開啓標準的ISO 8601格式:app

@Bean
public ObjectMapper serializingObjectMapper() {
    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS);
    objectMapper.registerModule(new JavaTimeModule());
    return objectMapper;
}

此時,咱們在訪問剛纔的接口,就再也不是數組類型了,同時對於Feign客戶端的調用也不會再出現上面的錯誤了。post

本文首發: http://blog.didispace.com/Spr...

代碼示例

本文的相關例子能夠查看下面倉庫中的Chapter3-1-7目錄:

Spring Booot 2.0 新特性詳解正在連載,點擊看看都有哪些解讀

相關文章
相關標籤/搜索