在 Java DTO 里把
`@JsonFormat` + `@JsonSerialize` + `@JsonDeserialize` 三个注解同时写在同一个 `LocalDate` 字段上,**绝大多数情况下是冗余的**,甚至可能因为配置不一致而互相覆盖,导致序列化/反序列化行为不符合预期。下面把原理、最佳实践和常见坑一次讲清。
------------------------------------------------
1. 每个注解到底干了什么
------------------------------------------------
| 注解 | 归属 | 作用时机 | 典型用途 |
|---|---|---|---|
| `@JsonFormat` | Jackson 内核 | 序列化 & 反序列化 | 指定“格式字符串”和时区,只对外层是 **时间/日期类型** 的字段生效。 |
| `@JsonSerialize` | Jackson 内核 | 仅序列化 | 完全自定义序列化逻辑,会 **覆盖** `@JsonFormat` 的序列化行为。 |
| `@JsonDeserialize` | Jackson 内核 | 仅反序列化 | 完全自定义反序列化逻辑,会 **覆盖** `@JsonFormat` 的反序列化行为。 |
------------------------------------------------
2. 一起写会发生什么
------------------------------------------------
- 如果只写 `@JsonFormat(pattern="yyyy-MM-dd", timezone="GMT+8")`,
Jackson 自带的 `LocalDate` 内置 (de)serializer 会识别该注解,按指定格式和时区工作,**不需要**额外写 `using = ...`。
- 一旦你又写了
`@JsonSerialize(using = LocalDateSerializer.class)`
`@JsonDeserialize(using = LocalDateDeserializer.class)`
就等于告诉 Jackson:“别用内置逻辑,改用我自己指定的 Serializer/Deserializer”。
而 `LocalDateSerializer` / `LocalDateDeserializer` 的源码里 **并没有去读 `@JsonFormat` 的 pattern 和 timezone**,它们只会走 JDK 默认的 `toString` / `parse` 逻辑(ISO 格式 `yyyy-MM-dd`)。
结果:
– 序列化:永远是 `yyyy-MM-dd`,**时区被忽略**(`LocalDate` 本身不带时区,所以 timezone 参数其实没意义)。
– 反序列化:只能接受 ISO 格式,**pattern 参数被忽略**。
------------------------------------------------
3. 结论:怎么写最干净
------------------------------------------------
1. **99% 场景**——只想固定格式、时区,**不要写后两个注解**:
```java
public class OrderDTO {
@JsonFormat(pattern = "yyyy-MM-dd", timezone = "GMT+8")
private LocalDate orderDate;
}
```
2. **1% 场景**——格式/逻辑太特殊,内置做不到,才自己写 Serializer/Deserializer,并且**不要再写 `@JsonFormat`**,避免歧义:
```java
public class OrderDTO {
@JsonSerialize(using = MyCustomLocalDateSerializer.class)
@JsonDeserialize(using = MyCustomLocalDateDeserializer.class)
private LocalDate orderDate;
}
```
------------------------------------------------
4. 常见疑问
------------------------------------------------
Q1. 时区对 `LocalDate` 有用吗?
→ 没用。`LocalDate` 只有年月日,没有时分区,因此 `@JsonFormat(timezone = "...")` 对 `LocalDate` 字段**实际不起作用**;不过写上也不会报错,只是给人看。
Q2. 我想统一全局格式,不想每个字段写 `@JsonFormat`?
→ 在 `application.yml`(Spring Boot)里配:
```yaml
spring:
jackson:
serialization:
write-dates-as-timestamps: false
deserialization:
accept-single-value-as-array: true
date-format: yyyy-MM-dd # 对 Date 类型生效
```
或者注册一个 `JavaTimeModule` 并自定义 `LocalDate` 的 (de)serializer,一次性全局生效。
------------------------------------------------
一句话总结
**“要么只写 `@JsonFormat`,要么只写 `@JsonSerialize`/`@JsonDeserialize`,不要三者一起堆在 `LocalDate` 上。”**
- 环境配置
- window怎么配置java环境变量?
- SQL学习
- 字段类型
- 1、 所有可声明的字段类型
- 2、常用的可声明的字段类型
- 创建表和表内字段
- 1、整个创建表语句
- 2、设置id主键
- 3、设置业务主键,唯一索引
- 4、设置业务字段
- 5、普通索引
- 6、唯一索引与普通索引的区别
- 7、表的引擎、字符集、排序规则和注释的设置
- Java基础语法
- 数据类型
- Java中的数据类型
- LocalDate
- 常用的数据类型
- Java 常用数据类型方法
- Java中数组、list、Map、HashMap
- 如何用Map来优化那些复杂的“双重 for 循环”查询?
- Java 8 中steam()操作流
- Java中HashMap和JSON
- Java中的JSONObject
- 工具类
- Java 常用工具类
- Arrays工具类
- Java 时间工具类
- 泛型
- Java中泛型概念
- Java中的泛型容器
- 泛型参数与函数参数的区别
- 推断出泛型实参
- Lambda表达式
- 数据分层
- 异常处理
- Java8 异常处理类总结表
- MyBatis-Plus 常用异常类总结表
- Java高级特性
- Maven
- jib-maven-plugin
- 什么是Spring Boot 的 parent pom?
- maven中各个生命周期的含义
- Spring Boot
- maven与spring boot 的关系
- Java中的连接池
- Spring JDBC
- Spring JDBC的概念
- JdbcTemplate常用的方法
- Spring中Bean的概念
- Spring中的抽象,通俗解释一下
- Spring中的事物
- Spring中的事物,通俗解释一下
- Spring中的事物抽象,常见的有哪些,列举一下
- Spring中常用的事物场景有哪些,列举一下
- Spring事务管理有哪些注解?
- Spring中使用事物处理订单的案例,列举说明一下
- Spring中声明式事务、分布式事务以及编程式事务的区别,列举一下
- 配置文件
- application-properties配置文件
- Spring Boot 的启动
- spring boot项目如何启动?
- 列举一下Spring Boot的启动过程
- SpringApplication.run方法
- Spring Boot 启动时有哪些接口?
- CommandLineRunner
- Spring Boot 的常用注解
- 系统注解
- 表格:系统注解
- @Override
- @Deprecated
- @SuppressWarnnings
- 使用在类名上的注解
- 表格:使用在类名上的注解
- @RestController
- @Controller
- @Service
- @Repository
- @Component
- @Configuration
- @Resource
- @Autowired
- @RequestMapping
- @GetMapping
- @PostMapping
- @Transactional
- @Qualifier
- 使用在方法上的注解
- 表格:使用在方法上的注解
- @RequestBody
- @PathVariable
- @Bean
- @ResponseBody
- @PreAuthorize
- 其他常用注解
- 表格:其他常用注解
- @EnableAutoConfiguration
- @SpringBootApplication
- @EnableScheduling
- @EnableAsync
- @ComponentScan
- @Aspec
- @ControllerAdvice
- @ExceptionHandler
- @Value
- @ConfigurationProperties
- @EnableConfigurationProperties
- @MapperScan
- @ApiOperation
- @Produces
- Validator验证的常用注解
- spring IoC容器
- Spring IoC容器依赖注入实现方式
- 常用依赖
- RESTEasy
- resteasy简介
- RESTEasy框架(依赖)的功能和常用注解
- MyBatis
- 简介
- paginationInterceptor
- @TableName
- @TableId
- @Param
- MyBatis-Plus
- MyBatis-Plus简介
- MyBatis-Plus的工具类
- Mybatis-Plus扩展的工具类和方法
- MyBatis-Plus中最常用的工具类方法
- MyBatis-Plus 中最常用的 4 大核心工具类
- Wrapper条件构造器
- Wrapper条件构造器详解
- Wrapper条件构造器eq等方法的参数说明
- LambdaQueryWrapper与QueryWrapper
- 日期格式是否必须转换
- Lombok
- Lombok作用详解
- @Data
- @Slf4j
- @Builder
- @EqualsAndHashCode
- @Accessors
- Jackson
- Jackson简介
- @JsonFormat
- Jackson高效地在 HashMap 和 JSON 字符串之间进行相互转换
- Hutool
- Hutool简介
- hutool依赖常用的方法
- fastjson2
- fastjson2简介
- UrlBasedCorsConfigurationSource
- 生态相关
- JBoss 社区
- 支付系统
- 1. 初始化mysql数据库流程
- 2. 初始化redis数据库的流程
- 3. 初始化rabbitmq服务
- 环球置业
- 1.模块目录结构分析
- 2. DTO(数据传输层)的核心作用
- 3. VO(视图对象层)
- 4. VO(视图对象层)和 DTO 数据传输层 的区别
