今天遇到了一个坑爹的问题,当用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个小时。所以就形成了上面所说的那种坑的情况。
解决方案:
在你每个日期类型的字段上的 @JsonFormat 加上属性 timezone="GMT+8"
@NotNull
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",timezone="GMT+8")
@JsonProperty("start_time")
private Date startTime;
这种解决方法会让你在每个日期类型上都加上这么一个属性,比较繁琐。
还有一种操作,只需要配置一个bean就行,代码如下:
@Bean
public Jackson2ObjectMapperBuilderCustomizer jacksonObjectMapperCustomization() {
return jacksonObjectMapperBuilder ->
jacksonObjectMapperBuilder.timeZone(TimeZone.getTimeZone("GMT+8"));
}