問題】
Demo:
LocalDateTime dt = LocalDateTime.now(); ObjectMapper mapper = new ObjectMapper(); try { String json = mapper.writeValueAsString(dt); System.out.println(json); } catch (JsonProcessingException e) { e.printStackTrace(); }
Jackson默認序列化會將LocalDateTime序列化成:
{"dayOfMonth":15,"dayOfWeek":"SUNDAY","dayOfYear":258,"hour":14,"minute":6,"month":"SEPTEMBER","monthValue":9,"nano":359000000,"second":55,"year":2019,"chronology":{"id":"ISO","calendarType":"iso8601"}}
並且該字符串反序列化成LocalDateTime會報錯:
Cannot construct instance of `java.time.LocalDateTime`
Demo:
String json = "{\"dayOfMonth\":15,\"dayOfWeek\":\"SUNDAY\",\"dayOfYear\":258,\"hour\":14,\"minute\":2,\"month\":\"SEPTEMBER\",\"monthValue\":9,\"nano\":71000000,\"second\":46,\"year\":2019,\"chronology\":{\"id\":\"ISO\",\"calendarType\":\"iso8601\"}}"; ObjectMapper mapper = new ObjectMapper(); try { LocalDateTime dt = mapper.readValue(json, LocalDateTime.class); System.out.println(dt); } catch (IOException e) { e.printStackTrace(); }
就算加了:
mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS); mapper.registerModule(new JavaTimeModule());
也會報如下錯誤:
com.fasterxml.jackson.databind.exc.MismatchedInputException: Expected array or string.
Demo:
String json = "{\"dayOfMonth\":15,\"dayOfWeek\":\"SUNDAY\",\"dayOfYear\":258,\"hour\":14,\"minute\":2,\"month\":\"SEPTEMBER\",\"monthValue\":9,\"nano\":71000000,\"second\":46,\"year\":2019,\"chronology\":{\"id\":\"ISO\",\"calendarType\":\"iso8601\"}}"; ObjectMapper mapper = new ObjectMapper(); try { mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS); mapper.registerModule(new JavaTimeModule()); LocalDateTime dt = mapper.readValue(json, LocalDateTime.class); System.out.println(dt); } catch (IOException e) { e.printStackTrace(); }
【解決方案】
思路:
改變jackson內部序列化和反序列化LocalDateTime的方式,序列化時轉換為時間戳,反序列化時將時間戳轉換為LocalDateTime即可。
代碼:
創建兩個類,分別繼承JsonSerializer和JsonDeserializer
//時間序列化時變為時間戳 public class LocalDateTimeSerializer extends JsonSerializer<LocalDateTime> { @Override public void serialize(LocalDateTime localDateTime, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException { jsonGenerator.writeNumber(localDateTime.toInstant(ZoneOffset.ofHours(8)).toEpochMilli()); } }
//時間戳反序列化時間 public class LocalDateTimeDeserializer extends JsonDeserializer<LocalDateTime> { @Override public LocalDateTime deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException, JsonProcessingException { Long timestamp = jsonParser.getLongValue(); return LocalDateTime.ofEpochSecond(timestamp / 1000, 0, ZoneOffset.ofHours(8)); } }
使用方法(序列化添加如下3行紅色代碼)
LocalDateTime dt = LocalDateTime.now(); ObjectMapper mapper = new ObjectMapper(); try { JavaTimeModule timeModule = new JavaTimeModule(); timeModule.addSerializer(LocalDateTime.class, new LocalDateTimeSerializer()); mapper.registerModule(timeModule); String json = mapper.writeValueAsString(dt); System.out.println(json); } catch (JsonProcessingException e) { e.printStackTrace(); }
* 經過此方式序列化得到的結果為:"1568528927273"
使用方法(反序列化添加如下3行紅色代碼)
String json = "1568528927273"; ObjectMapper mapper = new ObjectMapper(); try { JavaTimeModule timeModule = new JavaTimeModule(); timeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer()); mapper.registerModule(timeModule); LocalDateTime dt = mapper.readValue(json, LocalDateTime.class); System.out.println(dt); } catch (IOException e) { e.printStackTrace(); }
轉自https://www.cnblogs.com/yzeng/p/11522411.html