SpringBoot配置LocalDateTime转Long的自定义Converter不生效
SpringBoot配置LocalDateTime转Long的自定义Converter不生效
今天玩mybatis-plus时默认生成的entity的日期类型是LocalDateTime,但是在controller中返回时SpringBoot给转成了一个数组,前端格式化有点问题。并且当前端请求后端时,Long转LocalDateTime直接报错。
网上找了很多资料,比如重写extendMessageConverters,重写configureMessageConverters,把converters的index设置为0,把ObjectMapper注册为Bean,等等方式,全部宣告失败。
解决方式
直到我找了几乎1个小时解决方式没有找到,我决定去跟源码。然后我发现了objectMapper的一个叫做“忽略重复的模块注册”的配置 IGNORE_DUPLICATE_MODULE_REGISTRATIONS。
对,就是objectMapper.registerModule时的第一个判断条件。
于是,我把我重写extendMessageConverters的逻辑改成了这样
@Configuration
public class WebConfig implements WebMvcConfigurer {
@Override
public void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
ObjectMapper objectMapper = converter.getObjectMapper();
// 生成JSON时,将所有Date转换成Long
JavaTimeModule javaTimeModule = new JavaTimeModule();
javaTimeModule.addSerializer(LocalDateTime.class, new LocalDateTimeToLongSerializer());
javaTimeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeToLongDeserializer());
// 把“忽略重复的模块注册”禁用,否则下面的注册不生效
objectMapper.disable(MapperFeature.IGNORE_DUPLICATE_MODULE_REGISTRATIONS);
objectMapper.registerModule(javaTimeModule);
// 然后再设置为生效,避免被其他地方覆盖
objectMapper.enable(MapperFeature.IGNORE_DUPLICATE_MODULE_REGISTRATIONS);
// 设置格式化内容
converter.setObjectMapper(objectMapper);
converters.add(0, converter);
}
}
public class LocalDateTimeToLongDeserializer extends JsonDeserializer<LocalDateTime> {
@Override
public LocalDateTime deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException {
long timestamp = jsonParser.getValueAsLong();
Instant instant = Instant.ofEpochMilli(timestamp);
return LocalDateTime.ofInstant(instant, ZoneId.systemDefault());
}
}
public class LocalDateTimeToLongSerializer extends JsonSerializer<LocalDateTime> {
@Override
public void serialize(LocalDateTime date, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException {
jsonGenerator.writeNumber(date.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli());
}
}
然后就好了!!
总结
事实证明,一味的百度、谷歌还是不能解决所有问题,平时多看看源码,比啥都强。