Jackson日期反序列化时区问题

今天遇到了一个坑爹的问题,当用Jackson对Json中的日期类型进行反序列化时,出现了日期错误的问题,当时写法是这样的:

@Data
@NoArgsConstructor
public class TotalByAccessIdRest {

    @NotNull
    @JsonProperty("access_id")
    private String accessId;

    @NotNull
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    @JsonProperty("start_time")
    private Date startTime;

    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    @JsonProperty("end_time")
    private Date endTime = new Date();
}

按平时来看,好像根本没有什么问题。但当你发送请求,比如你发送startTime或者endTime参数值为2018-04-13 00:00:00时间时,在Controller端实际你收到后的值就变成了2018-04-13 08:00:00,为什么会这样子呢?

让我们进入Jackson里 JsonFormat的源码看看:

    /**
     * Value that indicates that default {@link java.util.TimeZone}
     * (from deserialization or serialization context) should be used:
     * annotation does not define value to use.
     *

* NOTE: default here does NOT mean JVM defaults but Jackson databindings * default, usually UTC, but may be changed on ObjectMapper. */ public final static String DEFAULT_TIMEZONE = "##default";

会看到这样一段代码,那个注释已经说明默认情况下会将 时区设置为UTC ,Jackson反序列化时间类型的底层实际上调用的是Java的 SimpleDateFormat#parse() 方法, 而JVM中的时区则会根据你的操作系统来获取,所以JVM认为你的时区应该是 GMT+8 时区,而要将 UTC 时区的时间转成 GMT+8 时区的时间,就会将你传进来的时间+8个小时。所以就形成了上面所说的那种坑的情况。

解决方案:

  1. 在你每个日期类型的字段上的 @JsonFormat 加上属性 timezone="GMT+8"
    @NotNull
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
    @JsonProperty("start_time")
    private Date startTime;

这种解决方法会让你在每个日期类型上都加上这么一个属性,比较繁琐。

  1. 还有一种操作,只需要配置一个bean就行,代码如下:
    @Bean
    public Jackson2ObjectMapperBuilderCustomizer jacksonObjectMapperCustomization() {
        return jacksonObjectMapperBuilder ->
                jacksonObjectMapperBuilder.timeZone(TimeZone.getTimeZone("GMT+8"));
    }

你可能感兴趣的:(Jackson日期反序列化时区问题)