Cannot deserialize value of type `java.sql.Timestamp` from String “2022-07-08 16:53:40“: not a valid

在SpringBoot项目中遇到一个问题,当尝试将Timestamp类型的日期转换为字符串时,由于格式不匹配导致错误。解决方案是在application.yml配置Jackson的时区和日期格式,并在实体类中使用`@DateTimeFormat`注解指定日期格式。

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

Cannot deserialize value of type java.sql.Timestamp from String “2023-05-01”: not a valid representation (error: Failed to parse Date value ‘2023-05-01’: Unparseable date: “2023-05-01”)


问题描述

在做一个SpringBoot项目的时候遇到了在前端中,拿到时间Timestamp时间类型,数据库中对应的字段数据类datatime,在给这个对象封装为json格式存储在数据库中,就爆出了这个错误

Cannot deserialize value of type `java.sql.Timestamp` from String "2023-05-01": 
not a valid representation (error: Failed to parse Date value '2023-05-01': Unparseable date: "2023-05-01")

原因分析:

数据类型转换错误,错误的转换Date时间类型


解决方案:

在application.yml中添加

spring:
jackson:
time-zone: GMT+8
date-format: yyyy-MM-dd

在对应的类中加入
@DateTimeFormat(pattern = "yyyy-MM-dd")
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Note {
    private Integer id;
    private String noteName;
    private String noteContent;
    @DateTimeFormat(pattern = "yyyy-MM-dd")
    private Timestamp noteTime;
    private User user;
}

<think>我们正在解决一个具体问题:在Java中反序列化字符串"2025-08-11 16:53:58"到java.util.Date时出现错误。 错误信息类似于:Cannot deserialize value of type `java.util.Date` from String "2025-08-11 16:53:58": 期望的格式可能是"yyyy-MM-dd HH:mm:ss"(但实际没有指定格式,导致解析失败) 根据引用[3]中的错误信息,当字符串格式与Jackson默认期望的格式不匹配时,就会抛出异常。Jackson默认支持ISO8601格式(如"2024-04-09T16:00:00.000Z")和长整型时间戳。对于其他格式,需要明确指定。 解决方案: 1. 在对应的Date字段上使用@JsonFormat注解指定格式。 2. 配置全局的日期格式(如果多个字段使用相同格式)。 3. 使用自定义反序列化器(如引用[4]所示,适用于复杂情况)。 具体步骤: 方法一:使用@JsonFormat注解(适用于单个字段) 在DTO类的日期字段上添加注解,指定与字符串匹配的格式。 方法二:全局配置(适用于整个应用使用同一种日期格式) 在Spring Boot应用中,可以在application.properties中配置: spring.jackson.date-format=yyyy-MM-dd HH:mm:ss spring.jackson.time-zone=GMT+8 # 根据实际情况设置时区 注意:全局配置会作用于所有Date类型的字段,如果不同字段格式不同,则不能使用全局配置。 方法三:自定义反序列化器(如引用[4]所示,适用于需要特殊处理的情况) 创建一个自定义的反序列化器,然后在需要特殊处理的字段上使用@JsonDeserialize注解指定该反序列化器。 针对当前问题,字符串是"2025-08-11 16:53:58",格式为"yyyy-MM-dd HH:mm:ss",我们可以使用方法一或方法二。 下面详细说明方法一: 步骤: 1. 在DTO类的日期字段上添加注解: import java.util.Date; import com.fasterxml.jackson.annotation.JsonFormat; public class MyDto { // 其他字段... @JsonFormat(shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8") private Date myDateField; // getter和setter } 注意:timezone属性根据实际情况设置,例如如果时间字符串是东八区时间,则设置为"GMT+8"。 如果多个字段都使用相同的格式,使用方法二(全局配置)更简便,避免重复注解。 全局配置(在Spring Boot的application.properties中): spring.jackson.date-format=yyyy-MM-dd HH:mm:ss spring.jackson.time-zone=Asia/Shanghai # 或者GMT+8 但是,全局配置会改变所有Date类型的序列化和反序列化行为,包括可能存在的其他格式的日期字段。如果应用中有多种日期格式,则不能使用全局配置,而应该使用注解在字段上单独指定。 如果问题仍然存在,检查: - 字符串是否严格符合指定的格式(例如,月份和日期是否为两位,小时是否为24小时制等) - 时区设置是否正确 引用[1]和[3]中的错误都是因为格式不匹配,所以通过指定格式可以解决。 下面是一个完整的示例: DTO类: ``` public class Account { private String name; @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8") private Date birthday; // 省略getter和setter } ``` 当接收到JSON字符串如:{"name":"John", "birthday":"2025-08-11 16:53:58"}时,Jackson就会按照指定的格式解析。 如果不想修改DTO类,也可以使用全局配置(在配置类中): ``` @Configuration public class JacksonConfig { @Bean public Jackson2ObjectMapperBuilder jackson2ObjectMapperBuilder() { Jackson2ObjectMapperBuilder builder = new Jackson2ObjectMapperBuilder(); builder.dateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss")); builder.timeZone(TimeZone.getTimeZone("GMT+8")); return builder; } } ``` 或者使用application.properties(Spring Boot): spring.jackson.date-format=yyyy-MM-dd HH:mm:ss spring.jackson.time-zone=GMT+8 总结:通过指定日期格式,Jackson就能正确解析字符串到Date对象。 相关问题: 1. 如何在Jackson中处理多种不同的日期格式? 2. 使用@JsonFormat注解时,时区设置不正确会导致什么问题? 3. Jackson除了支持Date类型,还支持哪些日期时间类型(如Java8的LocalDateTime)? 4. 如何将Date类型按照指定格式序列化为字符串?</think>### 解决 Java 日期反序列化错误:`Cannot deserialize value of type java.util.Date from String` #### 问题分析 当 Jackson 尝试将字符串 `"2025-08-11 16:53:58"` 反序列化为 `java.util.Date` 时,会抛出异常,因为: 1. **格式不匹配**:字符串格式为 `yyyy-MM-dd HH:mm:ss`,但 Jackson 默认仅支持 ISO 8601 格式(如 `"2024-04-09T16:00:00.000Z"`)[^1][^3] 2. **缺少时区信息**:字符串未包含时区(如 `+0800` 或 `Z`) 3. **未明确指定格式**:未告知 Jackson 如何解析此格式的字符串 --- ### 解决方案(3种方法) #### 方法 1:使用 `@JsonFormat` 注解(推荐) 在 DTO 类的日期字段上指定格式: ```java import com.fasterxml.jackson.annotation.JsonFormat; import java.util.Date; public class YourDTO { // 指定格式和时区 @JsonFormat( shape = JsonFormat.Shape.STRING, pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8" ) private Date yourDateField; // getter/setter } ``` **说明**: - `pattern`:必须与输入字符串格式**完全一致** - `timezone`:强制指定时区(东八区示例) #### 方法 2:全局配置日期格式 在 `application.properties` 中配置: ```properties # 指定默认日期格式 spring.jackson.date-format=yyyy-MM-dd HH:mm:ss # 指定默认时区(GMT+8) spring.jackson.time-zone=GMT+8 ``` #### 方法 3:自定义反序列化器(复杂场景) 当需要处理多种格式时: 1. 创建自定义反序列化器 ```java import com.fasterxml.jackson.core.JsonParser; import com.fasterxml.jackson.databind.DeserializationContext; import com.fasterxml.jackson.databind.JsonDeserializer; import java.io.IOException; import java.text.SimpleDateFormat; import java.util.Date; public class CustomDateDeserializer extends JsonDeserializer<Date> { private static final SimpleDateFormat FORMATTER = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); static { FORMATTER.setTimeZone(TimeZone.getTimeZone("GMT+8")); } @Override public Date deserialize(JsonParser p, DeserializationContext ctx) throws IOException { String dateStr = p.getText(); try { return FORMATTER.parse(dateStr); } catch (Exception e) { throw new IOException("Failed to parse date: " + dateStr, e); } } } ``` 2. 在 DTO 字段上应用 ```java import com.fasterxml.jackson.databind.annotation.JsonDeserialize; public class YourDTO { @JsonDeserialize(using = CustomDateDeserializer.class) private Date yourDateField; } ``` --- ### 关键注意事项 1. **格式必须严格匹配**: - 输入 `"2025-08-11 16:53:58"` → 格式 `yyyy-MM-dd HH:mm:ss` - 使用空格而非 `T` 分隔日期和时间[^1][^3] 2. **时区问题**: - 未指定时区时,Jackson 默认使用系统时区 - 跨时区系统需显式设置 `timezone` 参数 3. **毫秒处理**: - 若字符串包含毫秒(如 `"2025-08-11 16:53:58.123"`),需修改格式为 `yyyy-MM-dd HH:mm:ss.SSS` --- ### 验证测试 ```java // 测试反序列化 String json = "{\"yourDateField\":\"2025-08-11 16:53:58\"}"; ObjectMapper mapper = new ObjectMapper(); YourDTO dto = mapper.readValue(json, YourDTO.class); // 验证结果 SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); sdf.setTimeZone(TimeZone.getTimeZone("GMT+8")); System.out.println(sdf.format(dto.getYourDateField())); // 输出:2025-08-11 16:53:58 ``` --- ### 常见错误排查 | 错误现象 | 解决方案 | |---------|---------| | `Unparseable date` | 检查格式字符串是否与输入完全匹配 | | 时间偏移几小时 | 检查时区设置(如 `GMT+8` 或 `Asia/Shanghai`) | | 毫秒部分丢失 | 在格式中添加 `.SSS` 并确保输入有3位毫秒 | | 不支持24小时制 | 使用 `HH` 而非 `hh`(`HH`=24小时制) | --- ### 相关问题 1. 如何同时处理多种日期格式(如 `yyyy-MM-dd` 和 `yyyy/MM/dd HH:mm:ss`)?[^4] 2. 使用 `java.time.LocalDateTime` 替代 `java.util.Date` 有哪些优势? 3. 如何在 Spring Boot 中全局配置多时区日期处理? 4. Jackson 反序列化时如何自动兼容带 `T` 和不带 `T` 的 ISO 格式?[^1][^3]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值