Jackson序列化与反序列化LocalDateTime、TimeStamp 等时间类型参数报错

在使用Jackson处理包含LocalDateTime或Timestamp的时间类型参数时,由于日期格式不匹配导致反序列化错误。解决方案包括全局配置WebMvcConfigurer中的ObjectMapper,或者针对特定属性使用@JsonFormat注解指定日期格式。在某些情况下,application.properties配置文件中的date-format可能无法生效,需要在WebMvcConfigurer中直接设置MappingJackson2HttpMessageConverter。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

场景:
实体类Entity 部分属性:

 /**
     * 创建时间
     */
    @Column(name = "gmt_create")
    protected Timestamp gmtCreate;
    /**
     * 修改时间
     */
    @Column(name = "gmt_modified")
    protected Timestamp gmtModified;

controller层采用实体类接收参数:

    @PostMapping("/testFeign")
    public void testFeign(@RequestBody JourneyEntity journeyEntity) {
         //业务代码
    }

Postman 发起post请求:
参数如下:
{
“journeyId”:“fd96adc11d6d4e11a0c184297dbedc98”,
“name”:“测试版本下线”,
“version”:“3”,
“maxVersion”:“4”,
“audience”:“CROWD”,
“type”:“service”,
“gmtCreate”:“2020-05-27 10:59:40”,–字符串类型参数
“conflictResolution”:“concurrent”,
“layoutContent”:“hhhhh”,
“policyDelivery”:"{‘type’:‘IMMEDIATELY’}"
}

结果报错如下:

Caused by: com.fasterxml.jackson.databind.exc.InvalidFormatException: Cannot deserialize value of type `java.sql.Timestamp` from String "2020-05-27 10:59:40": not a valid representation (error: Failed to parse Date value '2020-05-27 10:59:40': Cannot parse date "2020-05-27 10:59:40": while it seems to fit format 'yyyy-MM-dd'T'HH:mm:ss.SSSZ', parsing fails (leniency? null))
 at [Source: (PushbackInputStream); line: 8, column: 14] (through reference chain: com.aliyun.opaas.journey.dal.entity.JourneyEntity["gmtCreate"])
	at com.fasterxml.jackson.databind.exc.InvalidFormatException.from(InvalidFormatException.java:67)
	at com.fasterxml.jackson.databind.DeserializationContext.weirdStringException(DeserializationContext.java:1549)
	at com.fasterxml.jackson.databind.DeserializationContext.handleWeirdStringValue(DeserializationContext.java:911)
	at com.fasterxml.jackson.databind.deser.std.StdDeserializer._parseDate(StdDeserializer.java:524)
	at com.fasterxml.jackson.databind.deser.std.StdDeserializer._parseDate(StdDeserializer.java:467)
	at com.fasterxml.jackson.databind.deser.std.DateDeserializers$DateBasedDeserializer._parseDate(DateDeserializers.java:195)
	at com.fasterxml.jackson.databind.deser.std.DateDeserializers$TimestampDeserializer.deserialize(DateDeserializers.java:335)
	at com.fasterxml.jackson.databind.deser.std.DateDeserializers$TimestampDeserializer.deserialize(DateDeserializers.java:320)
	at com.fasterxml.jackson.databind.deser.impl.MethodProperty.deserializeAndSet(MethodProperty.java:127)
	at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserializeFromObject(BeanDeserializer.java:369)
	at com.fasterxml.jackson.databind.deser.BeanDeserializer.deserialize(BeanDeserializer.java:159)
	at com.fasterxml.jackson.databind.ObjectMapper._readMapAndClose(ObjectMapper.java:4013)
	at com.fasterxml.jackson.databind.ObjectMapper.readValue(ObjectMapper.java:3084)
	at org.springframework.http.converter.json.AbstractJackson2HttpMessageConverter.readJavaType(AbstractJackson2HttpMessageConverter.java:239)
	... 66 more

异常处可明显看到,将String 反序列化成Timestamp类型时报错,反序列化用的格式是:
yyyy-MM-dd’T’HH:mm:ss.SSSZ,而String的格式是:yyyy-MM-dd HH:mm:ss

解决办法:

  1. 全局解决:
    设置WebMvcConfigurer 中的ObjectMapper的时间转换类型,并注入HttpMessageConverter中
@Configuration
public class WebConfig implements WebMvcConfigurer {
    @Autowired
    private Jackson2ObjectMapperBuilder jackson2ObjectMapperBuilder;

     @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        ObjectMapper mapper = jackson2ObjectMapperBuilder.build();
        mapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));
        converters.add(0, new MappingJackson2HttpMessageConverter(mapper));
    }
}
  1. 针对属性解决:
    添加注解@JsonFormat(pattern = “yyyy-MM-dd HH:mm:ss”, timezone=“GMT+8”)
    @Column(name = "gmt_create")
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss",  timezone="GMT+8")
    protected Timestamp gmtCreate;

这样可保证不会影响全局,只针对单个属性进行json转换

  1. 添加spring关于jackson的属性配置(试了后,我的环境上并没有生效,如有建议欢迎提出,一起成长)
    application.properties 配置文件添加
spring.jackson.date-format=yyyy-MM-dd HH:mm:ss
spring.jackson.time-zone=GMT+8
spring.jackson.default-property-inclusion=ALWAYS

不生效原因:
WebMvcConfigurer 配置类里已经设置了new MappingJackson2HttpMessageConverter()

    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        /*ObjectMapper mapper = jackson2ObjectMapperBuilder.build();
        mapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));*/
        /**
        *此处已经设置了MappingJackson2HttpMessageConverter jackson转换器,但是未设置任何转换格式,所以当需要反序列化String类型为Timestamp类型时,
        用的格式仍然Jackson默认的类型格式-yyyy-MM-dd'T'HH:mm:ss.SSSZ,
        导致反序列化失败,报错,而且此处设置便会覆盖配置文件的 jackson的配置项,配置一直不生效的原因就是这里
        **/
        converters.add(0, new MappingJackson2HttpMessageConverter());
    }

解决办法就是:有此处设置MappingJackson2HttpMessageConverter 时,只能由此处指定了date-format转换格式才能有效,如上代码中,ObjectMapper 来设置dateFormat

        ObjectMapper mapper = jackson2ObjectMapperBuilder.build();
        mapper.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));
        /**
        *此处已经设置了MappingJackson2HttpMessageConverter jackson转换器,但是未设置任何转换格式,所以当需要反序列化String类型为Timestamp类型时,
        用的格式仍然Jackson默认的类型格式-yyyy-MM-dd'T'HH:mm:ss.SSSZ,
        导致反序列化失败,报错,而且此处设置便会覆盖配置文件的 jackson的配置项,配置一直不生效的原因就是这里
        **/
        converters.add(0, new MappingJackson2HttpMessageConverter(mapper));
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值