SpringBoot 时间格式化
时间格式化在项目中使用频率是非常高的,当 API 接口返回结果,需要对其中某一个 date 字段属性进行特殊的格式化处理,通常会用到 SimpleDateFormat 工具处理。

  1. SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
  2. Date stationTime = dateFormat.parse(dateFormat.format(PayEndTime()));

可一旦处理的地方较多,不仅 CV 操作频繁,还产生很多重复臃肿的代码,而此时如果能将时间格式统一配置,就可以省下更多时间专注于业务开发了。
很多人觉得统一格式化时间很简单,像下边这样配置一下就可以了,但事实上这种方式只对 date 类型生效。

  1. spring.jackson.date-format=yyyy-MM-dd HH:mm:ss
  2. spring.jackson.time-zone=GMT+8

而很多项目中用到的时间和日期API 比较混乱, java.util.Datejava.util.Calendarjava.time LocalDateTime 都存在,所以全局时间格式化必须要同时兼容性新旧 API


  1. @Data
  2. public class OrderDTO {
  3. private LocalDateTime createTime;
  4. private Date updateTime;
  5. }

一、@JsonFormat 注解

@JsonFormat 注解方式严格意义上不能叫全局时间格式化,应该叫部分格式化,因为@JsonFormat 注解需要用在实体类的时间字段上,而只有使用相应的实体类,对应的字段才能进行格式化。

  1. @Data
  2. public class OrderDTO {
  3. @JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd")
  4. private LocalDateTime createTime;
  5. @JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss")
  6. private Date updateTime;
  7. }

字段加上 @JsonFormat 注解后,LocalDateTimeDate 时间格式化成功。

二、@JsonComponent 注解(推荐)

使用 @JsonFormat 注解并不能完全做到全局时间格式化,所以接下来使用 @JsonComponent 注解自定义一个全局格式化类,分别对 DateLocalDate 类型做格式化处理。

  1. @JsonComponent
  2. public class DateFormatConfig {
  3. @Value("${spring.jackson.date-format:yyyy-MM-dd HH:mm:ss}")
  4. private String pattern;
  5. /**
  6. * @description date 类型全局时间格式化
  7. * @date 2020/8/31 18:22
  8. */
  9. @Bean
  10. public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilder() {
  11. return builder -> {
  12. TimeZone tz = TimeZone.getTimeZone("UTC");
  13. DateFormat df = new SimpleDateFormat(pattern);
  14. df.setTimeZone(tz);
  15. builder.failOnEmptyBeans(false)
  16. .failOnUnknownProperties(false)
  17. .featuresToDisable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
  18. .dateFormat(df);
  19. };
  20. }
  21. /**
  22. * @description LocalDate 类型全局时间格式化
  23. * @date 2020/8/31 18:22
  24. */
  25. @Bean
  26. public LocalDateTimeSerializer localDateTimeDeserializer() {
  27. return new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(pattern));
  28. }
  29. @Bean
  30. public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilderCustomizer() {
  31. return builder -> builder.serializerByType(LocalDateTime.class, localDateTimeDeserializer());
  32. }
  33. }

DateLocalDate 两种时间类型格式化成功,此种方式有效。
实际开发中如果有个字段不想用全局格式化设置的时间样式,想自定义格式怎么办?
那就需要和 @JsonFormat 注解配合使用了。

  1. @Data
  2. public class OrderDTO {
  3. @JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd")
  4. private LocalDateTime createTime;
  5. @JsonFormat(locale = "zh", timezone = "GMT+8", pattern = "yyyy-MM-dd")
  6. private Date updateTime;
  7. }

@JsonFormat 注解的优先级比较高,会以 @JsonFormat 注解的时间格式为主。

三、@Configuration 注解

这种全局配置的实现方式与上边的效果是一样的。

注意:在使用此种配置后,字段手动配置@JsonFormat 注解将不再生效。

  1. @Configuration
  2. public class DateFormatConfig2 {
  3. @Value("${spring.jackson.date-format:yyyy-MM-dd HH:mm:ss}")
  4. private String pattern;
  5. public static DateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
  6. @Bean
  7. @Primary
  8. public ObjectMapper serializingObjectMapper() {
  9. ObjectMapper objectMapper = new ObjectMapper();
  10. JavaTimeModule javaTimeModule = new JavaTimeModule();
  11. javaTimeModule.addSerializer(LocalDateTime.class, new LocalDateTimeSerializer());
  12. javaTimeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer());
  13. objectMapper.registerModule(javaTimeModule);
  14. return objectMapper;
  15. }
  16. /**
  17. * @author fcant
  18. * @description Date 时间类型装换
  19. * @date 2020/9/1 17:25
  20. */
  21. @Component
  22. public class DateSerializer extends JsonSerializer<Date> {
  23. @Override
  24. public void serialize(Date date, JsonGenerator gen, SerializerProvider provider) throws IOException {
  25. String formattedDate = dateFormat.format(date);
  26. gen.writeString(formattedDate);
  27. }
  28. }
  29. /**
  30. * @author fcant
  31. * @description Date 时间类型装换
  32. * @date 2020/9/1 17:25
  33. */
  34. @Component
  35. public class DateDeserializer extends JsonDeserializer<Date> {
  36. @Override
  37. public Date deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException {
  38. try {
  39. return dateFormat.parse(jsonParser.getValueAsString());
  40. } catch (ParseException e) {
  41. throw new RuntimeException("Could not parse date", e);
  42. }
  43. }
  44. }
  45. /**
  46. * @author fcant
  47. * @description LocalDate 时间类型装换
  48. * @date 2020/9/1 17:25
  49. */
  50. public class LocalDateTimeSerializer extends JsonSerializer<LocalDateTime> {
  51. @Override
  52. public void serialize(LocalDateTime value, JsonGenerator gen, SerializerProvider serializers) throws IOException {
  53. gen.writeString(value.format(DateTimeFormatter.ofPattern(pattern)));
  54. }
  55. }
  56. /**
  57. * @author fcant
  58. * @description LocalDate 时间类型装换
  59. * @date 2020/9/1 17:25
  60. */
  61. public class LocalDateTimeDeserializer extends JsonDeserializer<LocalDateTime> {
  62. @Override
  63. public LocalDateTime deserialize(JsonParser p, DeserializationContext deserializationContext) throws IOException {
  64. return LocalDateTime.parse(p.getValueAsString(), DateTimeFormatter.ofPattern(pattern));
  65. }
  66. }
  67. }