背景1.1 问题Spring Boot 在处理对象的序列化和反序列化时,默认使用框架自带的JackSon配置。使用框架默认的,通常会面临如下问题: Date返回日期格式(建议不使用Date,但老项目要兼容
- Date返回日期格式(建议不使用Date,但老项目要兼容),带有T,如 2018-05-15T24:59:59:
- LocalDate返回日期对象为数组(框架中继承了 WebMvcConfigurationSupport);
- LocalDateTime时间转换失败等;
- 定义了日期类型,如LocalDate,前端对接时(post/get),如果传入日期字符串("2022-05-05"),会报String 转换为LocalDate失败;
- 返回long型数据,前端js存在精度问题,需做转换;
- 一些特殊对象要做业务特殊转换,如加解密等;
- 针对特殊的具体对象,在对象上面使用注解,如:
@JsonSerialize(using= JsonDateSerializer.class) private Date taskEndTime; @ApiModelProperty(value = "检查日期") @JsonFormat(pattern = "yyyy-MM-dd") private LocalDate checkDate;
- 重新实现WebMvcConfigurer接口,自定义JackSon配置。
- 继承 WebMvcConfigurationSupport类,自定义JackSon配置。
- 方案1的模式,在对应变量上加上注解,是可以解决问题,但是严重编码重复,不优雅;
- 实现WebMvcConfigurer接口与继承WebMvcConfigurationSupport类,是Spring Boot提供开发者做统全局配置类的两种模式,注意两种模式的差异,详情查看后续章节介绍(两种不同的模式,使用不当时,就会出现配置不生效的情况);
- JackSon配置说明

//在反序列化时忽略在 json 中存在但 Java 对象不存在的属性 mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); //在序列化时日期格式默认为 yyyy-MM-dd'T'HH:mm:ss.SSSZ ,比如如果一个类中有private Date date;这种日期属性,序列化后为:{"date" : 1413800730456},若不为true,则为{"date" : "2014-10-20T10:26:06.604+0000"} mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS,false); //在序列化时忽略值为 null 的属性 mapper.setSerializationInclusion(Include.NON_NULL); //忽略值为默认值的属性 mapper.setDefaultPropertyInclusion(Include.NON_DEFAULT); // 美化输出 mapper.enable(SerializationFeature.INDENT_OUTPUT); // 允许序列化空的POJO类 // (否则会抛出异常) mapper.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS); // 把java.util.Date, Calendar输出为数字(时间戳) mapper.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS); // 在遇到未知属性的时候不抛出异常 mapper.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES); // 强制JSON 空字符串("")转换为null对象值: mapper.enable(DeserializationFeature.ACCEPT_EMPTY_STRING_AS_NULL_OBJECT); // 在JSON中允许C/C++ 样式的注释(非标准,默认禁用) mapper.configure(JsonParser.Feature.ALLOW_COMMENTS, true); // 允许没有引号的字段名(非标准) mapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true); // 允许单引号(非标准) mapper.configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true); // 强制转义非ASCII字符 mapper.configure(JsonGenerator.Feature.ESCAPE_NON_ASCII, true); // 将内容包裹为一个JSON属性,属性名由@JsonRootName注解指定 mapper.configure(SerializationFeature.WRAP_ROOT_VALUE, true); //序列化枚举是以toString()来输出,默认false,即默认以name()来输出 mapper.configure(SerializationFeature.WRITE_ENUMS_USING_TO_STRING,true); //序列化Map时对key进行排序操作,默认false mapper.configure(SerializationFeature.ORDER_MAP_ENTRIES_BY_KEYS,true); //序列化char[]时以json数组输出,默认false mapper.configure(SerializationFeature.WRITE_CHAR_ARRAYS_AS_JSON_ARRAYS,true);View Code
- 实现WebMvcConfigurer接口
/** * java 8 LocalDateTime转换器 * * @author wangling */ public class LocalDateTimeFormatter implements Formatter<LocalDateTime> { private final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); @Override public LocalDateTime parse(String text, Locale locale) throws ParseException { return LocalDateTime.parse(text, formatter); } @Override public String print(LocalDateTime object, Locale locale) { return formatter.format(object); } }编写LocalDate转换函数
/** * java 8 localDate转换器 * * @author wangling */ public class LocalDateFormatter implements Formatter<LocalDate> { private final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd"); @Override public LocalDate parse(String text, Locale locale) throws ParseException { return LocalDate.parse(text, formatter); } @Override public String print(LocalDate object, Locale locale) { return formatter.format(object); } }编写Jackson配置 编写一个自定义的ObjectMapper bean对象,设置优先级替换默认bean。
/** * 项目全局配置类 * * @author wangling * @date 2022/06/10 */ @Configuration @RequiredArgsConstructor public class WebConfig implements WebMvcConfigurer { @Override public void addFormatters(FormatterRegistry registry) { registry.addFormatterForFieldType(LocalDate.class, new LocalDateFormatter()); registry.addFormatterForFieldType(LocalDateTime.class, new LocalDateTimeFormatter()); } @Bean @Primary public ObjectMapper ObjectMapper() { String dateTimeFormat = "yyyy-MM-dd HH:mm:ss"; String dateFormat = "yyyy-MM-dd"; String timeFormat = "HH:mm:ss"; ObjectMapper objectMapper = new ObjectMapper(); objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false); objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); JavaTimeModule javaTimeModule = new JavaTimeModule(); // 序列化 javaTimeModule.addSerializer( LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(dateTimeFormat))); javaTimeModule.addSerializer( LocalDate.class, new LocalDateSerializer(DateTimeFormatter.ofPattern(dateFormat))); javaTimeModule.addSerializer( LocalTime.class, new LocalTimeSerializer(DateTimeFormatter.ofPattern(timeFormat))); javaTimeModule.addSerializer( Date.class, new DateSerializer(false, new SimpleDateFormat(dateTimeFormat))); // 反序列化 javaTimeModule.addDeserializer( LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern(dateTimeFormat))); javaTimeModule.addDeserializer( LocalDate.class, new LocalDateDeserializer(DateTimeFormatter.ofPattern(dateFormat))); javaTimeModule.addDeserializer( LocalTime.class, new LocalTimeDeserializer(DateTimeFormatter.ofPattern(timeFormat))); javaTimeModule.addDeserializer(Date.class, new DateDeserializers.DateDeserializer() { @SneakyThrows @Override public Date deserialize(JsonParser jsonParser, DeserializationContext dc) { String text = jsonParser.getText().trim(); SimpleDateFormat sdf = new SimpleDateFormat(dateTimeFormat); return sdf.parse(text); } }); javaTimeModule.addSerializer(Long.class, ToStringSerializer.instance); javaTimeModule.addSerializer(BigInteger.class, ToStringSerializer.instance); objectMapper.registerModule(javaTimeModule); return objectMapper; } }
- WebMvcConfigurationSupport类

/** * 项目全局配置类 * * @author wangling * @date 2022/06/10 */ @Configuration public class MvcInterceptorConfig extends WebMvcConfigurationSupport { @Override protected void addFormatters(FormatterRegistry registry) { // 用于get 全局格式化日期转换 registry.addFormatterForFieldType(LocalDate.class, new LocalDateFormatter()); registry.addFormatterForFieldType(LocalDateTime.class, new LocalDateTimeFormatter()); } @Override protected void extendMessageConverters(List<HttpMessageConverter<?>> converters) { // 代替框架默认的JackSon配置 用于post 全局格式化日期转换,long转字符串 MappingJackson2HttpMessageConverter jackson2HttpMessageConverter = new MappingJackson2HttpMessageConverter(); jackson2HttpMessageConverter.setObjectMapper(ObjectMapper()); // 基于顺序,先执行自定义的 converters.add(0, jackson2HttpMessageConverter); } private ObjectMapper ObjectMapper() { String dateTimeFormat = "yyyy-MM-dd HH:mm:ss"; String dateFormat = "yyyy-MM-dd"; String timeFormat = "HH:mm:ss"; ObjectMapper objectMapper = new ObjectMapper(); //忽略空Bean转json的错误 objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false); //忽略 在json字符串中存在,但是在对象中不存在对应属性的情况,防止错误。 // 例如json数据中多出字段,而对象中没有此字段。如果设置true,抛出异常,因为字段不对应;false则忽略多出的字段,默认值为null,将其他字段反序列化成功 objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); JavaTimeModule javaTimeModule = new JavaTimeModule(); // 序列化 javaTimeModule.addSerializer( LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(dateTimeFormat))); javaTimeModule.addSerializer( LocalDate.class, new LocalDateSerializer(DateTimeFormatter.ofPattern(dateFormat))); javaTimeModule.addSerializer( LocalTime.class, new LocalTimeSerializer(DateTimeFormatter.ofPattern(timeFormat))); javaTimeModule.addSerializer( Date.class, new DateSerializer(false, new SimpleDateFormat(dateTimeFormat))); // 反序列化 javaTimeModule.addDeserializer( LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern(dateTimeFormat))); javaTimeModule.addDeserializer( LocalDate.class, new LocalDateDeserializer(DateTimeFormatter.ofPattern(dateFormat))); javaTimeModule.addDeserializer( LocalTime.class, new LocalTimeDeserializer(DateTimeFormatter.ofPattern(timeFormat))); javaTimeModule.addDeserializer(Date.class, new DateDeserializers.DateDeserializer() { @SneakyThrows @Override public Date deserialize(JsonParser jsonParser, DeserializationContext dc) { String text = jsonParser.getText().trim(); SimpleDateFormat sdf = new SimpleDateFormat(dateTimeFormat); return sdf.parse(text); } }); javaTimeModule.addSerializer(Long.class, ToStringSerializer.instance); javaTimeModule.addSerializer(BigInteger.class, ToStringSerializer.instance); objectMapper.registerModule(javaTimeModule); return objectMapper; } }View Code WebMvcConfigurer与WebMvcConfigurationSupport相关知识点
- 基础知识点
WebMvcConfigurer
接口提供了很多方法让开发者来定制SpringMVC的配置。
WebMvcConfigurationSupport implements ApplicationContextAware, ServletContextAware。支持的自定义的配置更多更全,WebMvcConfigurerAdapter有的方法,这个类也都有。该类注释内容翻译:这是提供MVC Java config 背后配置的主要类。 通常是通过将@EnableWebMvc添加到应用程序的@Configuration类中来导入的。 另一个更高级的选择是直接从此类扩展并在需要时重写方法,记住子类要添加@Configuration,重写带有@Bean的方法也要加上@Bean。
- 使用注意事项
- 实现WebMvcConfigurer: 不会覆盖WebMvcAutoConfiguration的配置
- 实现WebMvcConfigurer+注解@EnableWebMvc:会覆盖WebMvcAutoConfiguration的配置
- 继承WebMvcConfigurationSupport:会覆盖WebMvcAutoConfiguration的配置
- 继承DelegatingWebMvcConfiguration:会覆盖WebMvcAutoConfiguration的配置
- 推荐使用模式
- 非必要,最好避免WebMvcConfigurer,WebMvcConfigurationSupport在一个项目中同时使用;
- 出于安全性拦截配置,建议项目采用WebMvcConfigurer接口的方式做全局配置;
- 日期,时间等建议使用LocalDate,替换历史的Date数据类型;
微信赞赏
支付宝赞赏