当前位置: 首页 > article >正文

别再乱用@DateTimeFormat了!Spring Boot中处理前端日期传参的3种正确姿势(附Postman测试脚本)

别再乱用DateTimeFormat了Spring Boot中处理前端日期传参的3种正确姿势附Postman测试脚本最近在团队Code Review时发现不少同事在处理日期参数时都存在一个共性误区——把DateTimeFormat和JsonFormat混为一谈。这直接导致接口出现各种灵异现象明明注解加上了参数却死活解析失败测试环境好好的一到生产环境就报错。今天我们就来彻底解决这个痛点让你不再为日期传参掉头发。1. 为什么你的日期注解总是不生效先来看两个真实案例// 案例1URL传参失效 GetMapping(/orders) public ListOrder getOrders(RequestParam DateTimeFormat(patternyyyy-MM-dd) Date startDate) { // 实际接收到的startDate总是null } // 案例2JSON传参报错 PostMapping(/users) public User createUser(RequestBody UserDTO user) { // 报错Cannot deserialize value of type java.time.LocalDateTime }根本原因在于注解的场景错配。DateTimeFormat和JsonFormat虽然都能处理日期但它们的生效场景完全不同注解适用场景处理方向底层实现DateTimeFormatURL参数、表单数据请求参数解析Spring转换器JsonFormatJSON请求体/响应体序列化/反序列化Jackson库关键区别DateTimeFormat是Spring层面的参数转换而JsonFormat是JSON序列化/反序列化行为2. 三种场景下的正确解决方案2.1 URL查询参数处理当日期作为URL参数时如/api/orders?date2023-05-20必须使用DateTimeFormatGetMapping(/orders) public ResponseEntity? getOrders( RequestParam DateTimeFormat(pattern yyyy-MM-dd) LocalDate orderDate) { // 正确解析URL中的日期参数 }常见坑点忘记添加RequestParam注解使用了Java 8日期类型但未配置spring.mvc.format.date建议显式指定pattern时区未统一可通过spring.jackson.time-zoneGMT8全局配置2.2 JSON请求体处理对于POST/PUT请求中的JSON日期字段应该使用JsonFormatData public class OrderDTO { JsonFormat(pattern yyyy-MM-dd HH:mm:ss, timezone GMT8) private LocalDateTime createTime; } PostMapping(/orders) public Order createOrder(RequestBody OrderDTO order) { // 能正确反序列化JSON中的日期字符串 }关键配置# application.properties spring.jackson.date-formatyyyy-MM-dd HH:mm:ss spring.jackson.time-zoneGMT8 spring.jackson.serialization.write-dates-as-timestampsfalse2.3 混合场景处理当同一个接口既需要接收URL参数又要处理JSON体时PutMapping(/orders/{id}) public Order updateOrder( PathVariable Long id, RequestParam DateTimeFormat(iso DateTimeFormat.ISO.DATE) LocalDate auditDate, RequestBody OrderUpdateDTO updateDTO) { // URL参数和JSON体中的日期都能正确解析 }3. 完整实战从Controller到Postman测试3.1 完整代码示例RestController RequestMapping(/api) public class OrderController { // 场景1URL日期参数 GetMapping(/orders) public PageOrder queryOrders( RequestParam DateTimeFormat(patternyyyy-MM-dd) LocalDate startDate, RequestParam DateTimeFormat(patternyyyy-MM-dd) LocalDate endDate) { // 查询逻辑 } // 场景2JSON日期字段 PostMapping(/orders) public Order createOrder(RequestBody OrderCreateDTO createDTO) { // 创建逻辑 } // 场景3混合参数 PutMapping(/orders/{id}) public Order updateOrder( PathVariable String id, RequestParam DateTimeFormat(patternyyyy-MM-dd) LocalDate updateDate, RequestBody OrderUpdateDTO updateDTO) { // 更新逻辑 } } // DTO定义 Data public class OrderCreateDTO { JsonFormat(pattern yyyy-MM-dd HH:mm:ss) private LocalDateTime createTime; private String orderNumber; } Data public class OrderUpdateDTO { JsonFormat(pattern yyyy-MM-dd) private LocalDate deliveryDate; }3.2 Postman测试集合1. 测试URL参数接口GET /api/orders?startDate2023-05-01endDate2023-05-312. 测试JSON体接口POST /api/orders Body (raw-JSON): { createTime: 2023-05-20 14:30:00, orderNumber: ORD123456 }3. 测试混合参数接口PUT /api/orders/123?updateDate2023-05-15 Body (raw-JSON): { deliveryDate: 2023-05-25 }专业建议将测试用例保存为Postman Collection团队共享4. 高级技巧与避坑指南4.1 全局配置 vs 局部注解最佳实践组合# 全局默认配置 spring.mvc.format.dateyyyy-MM-dd spring.jackson.date-formatyyyy-MM-dd HH:mm:ss spring.jackson.time-zoneGMT8// 针对特殊字段单独配置 public class SpecialDTO { JsonFormat(pattern yyyy/MM/dd) // 覆盖全局配置 private LocalDate specialDate; }4.2 时区问题的终极解决方案前端统一传UTC时间戳推荐方案后端使用Instant类型接收展示时根据用户时区转换JsonFormat(pattern yyyy-MM-dd HH:mm:ss) private Instant createTime; // 存储和传输都使用UTC4.3 验证与异常处理添加参数验证GetMapping(/orders) public ResponseEntity? getOrders( RequestParam DateTimeFormat(pattern yyyy-MM-dd) FutureOrPresent // 必须是非过去日期 LocalDate date) { // ... }统一异常处理ControllerAdvice public class GlobalExceptionHandler { ExceptionHandler(MethodArgumentTypeMismatchException.class) public ResponseEntity? handleTypeMismatch() { return ResponseEntity.badRequest().body(日期格式错误请使用yyyy-MM-dd格式); } }5. 性能优化建议避免频繁创建SimpleDateFormat// 错误做法每次调用都新建实例 new SimpleDateFormat(yyyy-MM-dd).parse(dateStr); // 正确做法使用ThreadLocal缓存 private static final ThreadLocalDateFormat df ThreadLocal.withInitial(() - new SimpleDateFormat(yyyy-MM-dd));对于高并发接口建议使用时间戳代替字符串传输在Nginx层做日期格式校验对非法参数快速失败Jackson性能调优ObjectMapper mapper new ObjectMapper(); mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false); mapper.registerModule(new JavaTimeModule()); // 必须注册Java8日期模块最后分享一个真实踩坑案例我们曾经因为JsonFormat的timezone配置与数据库时区不一致导致每天凌晨1-2点的订单时间全部错乱。解决方案很简单——所有系统统一使用UTC时间只在展示层做时区转换。

相关文章:

别再乱用@DateTimeFormat了!Spring Boot中处理前端日期传参的3种正确姿势(附Postman测试脚本)

别再乱用DateTimeFormat了!Spring Boot中处理前端日期传参的3种正确姿势(附Postman测试脚本) 最近在团队Code Review时,发现不少同事在处理日期参数时都存在一个共性误区——把DateTimeFormat和JsonFormat混为一谈。这直接导致接口…...

保姆级教程:在Ubuntu 20.04上搞定RoboSense 16线雷达驱动与点云格式转换(附编译避坑指南)

跨平台LiDAR数据融合实战:Ubuntu 20.04/22.04下的RoboSense-Velodyne点云转换全解析 当我们在多传感器融合项目中尝试整合不同品牌的激光雷达时,数据格式的差异往往会成为第一个"拦路虎"。最近在部署RoboSense 16线雷达时,我发现许…...

从零到一:用VuePress/Hexo搭建技术博客时,你必须搞懂的SEO配置(附完整代码)

从零到一:用VuePress/Hexo搭建技术博客时,你必须搞懂的SEO配置(附完整代码) 技术博客不仅是开发者记录学习历程的载体,更是个人品牌的重要展示窗口。但很多开发者发现,即使内容优质,博客流量依然…...

Azure AI 模型费用精细统计实战 — Cost Management API 按资源组/模型/Token 类型拆分

通过 Azure Cost Management API 精细统计 AI 模型(GPT/Embedding/Speech/Vision)的费用,按资源组、模型、Token 类型(Input/Output/Cached)拆分,支持第三方平台集成。 一、背景 Azure AI 费用在 Cost Management 中归类为 Foundry Models(不是 “Azure AI Services”)…...

酒店信息数据集,数据量1.1万条,包含多个字段,可以用于酒店评分/价格/销量预测大数据分析毕设

酒店信息数据集,数据量1.1万条,包含多个字段,可以用于酒店评分/价格/销量预测大数据分析毕设,具体字段如下:酒店ID 酒店名称 图片URL 推荐理由 星级代码 星级描述 评分 评分描述 评论标签 评论数量 历史消费人数 原价 …...

Qwen1.5-0.5B-Chat成本控制:低配服务器部署实战案例

Qwen1.5-0.5B-Chat成本控制:低配服务器部署实战案例 1. 项目背景与价值 在AI应用快速发展的今天,很多开发者和中小企业都面临一个现实问题:如何以最低成本获得可用的智能对话服务?大模型虽然效果惊艳,但对硬件要求高…...

别再死磕卡尔曼滤波了!用Python从零实现一个RBPF粒子滤波建图(附避坑指南)

从零实现RBPF粒子滤波建图:Python实战与避坑指南 在机器人定位与建图(SLAM)领域,粒子滤波方法因其对非线性系统的天然适应能力而备受青睐。但大多数教程停留在理论推导,真正动手实现时总会遇到各种"魔鬼细节"。本文将用Python带你完…...

手把手教你用奥比中光Gemini 335L和YOLOv8实现3D目标定位(附完整代码)

从2D到3D:基于Gemini 335L与YOLOv8的智能空间感知实战指南 当计算机视觉遇上深度感知,会碰撞出怎样的火花?想象一下,你的设备不仅能识别物体是什么,还能精确知道它离你有多远——这正是3D目标定位技术的魅力所在。本文…...

告别全局update!手把手教你写一个安全的UVM寄存器批量更新函数

告别全局update!手把手教你写一个安全的UVM寄存器批量更新函数 在SoC验证环境中,寄存器配置是最基础却最频繁的操作之一。每次看到验证工程师手动逐个调用set()和update()时,我总会想起自己刚入行时那段"复制粘贴到怀疑人生"的日子…...

告别临时目录!Petalinux 2020.1 如何直接引用本地Uboot和Kernel源码进行开发

告别临时目录!Petalinux 2020.1 直接引用本地Uboot和Kernel源码的工程实践 在嵌入式Linux开发领域,Xilinx的Petalinux工具链因其与Zynq系列SoC的深度整合而广受欢迎。然而,许多工程师在实际开发中都会遇到一个共同的痛点:默认的&q…...

告别DETR训练慢!手把手教你用Deformable DETR在COCO数据集上10倍速收敛

10倍速训练秘籍:Deformable DETR在COCO数据集上的实战优化指南 当DETR首次亮相时,其端到端的目标检测架构确实令人耳目一新。但很快,实践者们就发现了一个棘手的问题:训练周期长得令人难以忍受。想象一下,在COCO数据集…...

固体、液体、气体摆式倾斜传感器到底怎么选?一张图看懂工业、消费电子、车载应用差异

固体、液体、气体摆式倾斜传感器工程选型指南:从工业级到消费电子的全场景解析 在智能硬件和工业自动化领域,倾斜传感器的选型往往决定着整个系统的可靠性和精度表现。作为一名经历过多次选型踩坑的硬件工程师,我深刻理解在面对无人机姿态控制…...

从‘残影’到对齐:用Python+pyrealsense2彻底搞懂RealSense D435i的RGB与深度图融合

从‘残影’到对齐:用Pythonpyrealsense2彻底搞懂RealSense D435i的RGB与深度图融合 第一次用RealSense D435i采集深度图时,我盯着屏幕上那些诡异的黑色边缘发呆了半小时——这到底是硬件故障还是正常现象?直到把咖啡杯挪到不同距离反复测试&a…...

WindowsCleaner:专治C盘爆红,让Windows系统重获新生的智能清理神器

WindowsCleaner:专治C盘爆红,让Windows系统重获新生的智能清理神器 【免费下载链接】WindowsCleaner Windows Cleaner——专治C盘爆红及各种不服! 项目地址: https://gitcode.com/gh_mirrors/wi/WindowsCleaner 你的电脑是否也经历过这…...

保姆级教程:在Ubuntu 22.04上为全志D1s/F133 MangoPi-MQ编译Tina Linux(含SDK结构解析)

全志D1s/F133开发板Tina Linux编译实战指南:从环境搭建到固件烧录 第一次拿到MangoPi-MQ这块小巧的开发板时,我对着全志Tina Linux的SDK文档发了半小时呆——lichee目录是干什么的?device和package又有什么区别?为什么官方教程里的…...

从ZED2双目相机到VINS-Fusion:一个机器人开发者的完整环境搭建与避坑实录

从ZED2双目相机到VINS-Fusion:一个机器人开发者的完整环境搭建与避坑实录 作为一名长期从事机器人感知算法开发的工程师,我最近在将ZED2双目相机与VINS-Fusion系统整合时,经历了一段充满挑战但又收获颇丰的旅程。这篇文章将详细记录我从零开始…...

用NDT算法给双Velodyne VLP-16做“对齐手术”:从采集数据到获得外参的完整实战

双Velodyne VLP-16激光雷达外参标定实战:基于NDT算法的数据驱动方法 在自动驾驶和机器人感知领域,多激光雷达系统的外参标定是构建精确环境模型的关键步骤。本文将深入探讨如何利用NDT(正态分布变换)算法,实现双Velody…...

JDK1.8环境下的传统系统AI升级:忍者像素绘卷与Java老项目集成

JDK1.8环境下的传统系统AI升级:忍者像素绘卷与Java老项目集成 1. 老系统AI升级的痛点与机遇 很多企业还在使用JDK1.8这样的老版本Java环境运行核心业务系统。这些系统通常已经稳定运行多年,但面临智能化升级的需求。传统系统引入AI能力时,常…...

JAVA同城组局找搭子小程序开发源码uniapp代码片段

以下是基于uniapp开发同城组局小程序的Java后端与前端代码片段参考,分为核心功能模块和关键技术实现:后端Java代码(Spring Boot框架)用户模块RestController RequestMapping("/api/user") public class UserController …...

AI 入门 30 天挑战 - Day 15 费曼学习法版 - 目标检测基础

🌟 完整项目和代码 本教程是 AI 入门 30 天挑战 系列的一部分! 💻 GitHub 仓库: https://github.com/Lee985-cmd/AI-30-Day-Challenge📖 CSDN 专栏: https://blog.csdn.net/m0_67081842?typeblog⭐ 欢迎 Star 支持!…...

嵌入式——认识电子元器件——电阻系列

认识常见电子元器件——电阻介绍名词解释精度温漂噪声过载分类按制造工艺/外观结构分类碳膜电阻金属膜电阻金属氧化膜电阻绕线电阻水泥电阻(陶瓷绕线)贴片电阻(片式厚膜/薄膜)按功能/特性分类固定电阻可调电阻(电位器&…...

从UVM-1.2源码看PH_TIMEOUT:超时机制详解与自定义超时策略配置指南

UVM超时机制深度解析:从源码实现到定制化策略实战 在芯片验证领域,UVM框架的超时机制就像一位严格的监考老师,当测试用例执行时间超出预期时,它会果断终止仿真并抛出PH_TIMEOUT错误。这种看似"无情"的设计背后&#xff…...

2026年终极终端战争:Warp vs. Tabby vs. 文心终端——软件测试工程师的专业选型指南

测试工程师为何需要关注终端革命?在软件开发的生命周期中,测试环节正变得前所未有的复杂与高效。自动化测试、持续集成/持续部署(CI/CD)、云原生与容器化技术的普及,使得测试工程师的工作场景早已超越了简单的图形界面…...

Spring Boot 4.0 Agent集成实战:从字节码注入到可观测性闭环,3步实现零侵入监控升级

第一章:Spring Boot 4.0 Agent-Ready 架构演进与核心价值Spring Boot 4.0 标志着 JVM 应用可观测性与运行时增强能力的一次范式跃迁。其核心设计理念是原生支持 Java Agent 集成,不再将字节码增强视为“外部插件行为”,而是作为框架生命周期的…...

3分钟解锁艾尔登法环帧率限制:告别卡顿的终极完整指南

3分钟解锁艾尔登法环帧率限制:告别卡顿的终极完整指南 【免费下载链接】EldenRingFpsUnlockAndMore A small utility to remove frame rate limit, change FOV, add widescreen support and more for Elden Ring 项目地址: https://gitcode.com/gh_mirrors/el/Eld…...

远程工作骗局:隐形加班——软件测试从业者的专业困境与破局之道

在数字浪潮席卷全球的今天,远程办公、混合工作制已成为包括软件测试行业在内的许多技术领域的“新常态”。它许诺了时间自由、通勤解放与生活平衡,一时间风靡无数职场人。然而,在这看似美好的工作模式背后,一个日益严峻且极具隐蔽…...

Java Stream里的‘懒’与‘急’:从面试题‘peek()为何不生效’讲透流操作原理

Java Stream里的‘懒’与‘急’:从面试题‘peek()为何不生效’讲透流操作原理 在Java开发者的日常工作中,Stream API已经成为集合处理的标配工具。但你是否遇到过这样的场景:在peek()方法中修改了元素,最终collect()的结果却"…...

# WebGPU实战:从零构建高性能图形渲染管线(附完整代码与流程图)在现代Web应用中,**图形渲染性能

WebGPU实战:从零构建高性能图形渲染管线(附完整代码与流程图) 在现代Web应用中,图形渲染性能的提升已成为开发者关注的核心问题之一。随着浏览器对硬件加速能力的支持不断增强,WebGPU作为下一代Web图形API,…...

超市售货管理平台小程序(文档+源码)_kaic

第5章 系统实现编程人员在搭建的开发环境中,会让各种编程技术一起呈现出最终效果。本节就展示关键部分的页面效果。5.1 管理员功能实现5.1.1 商品管理图5.1 即为编码实现的商品管理界面,管理员在商品管理界面中可以对界面中显示,可以对商品信…...

m4s-converter:3分钟实现B站缓存视频永久保存的终极方案

m4s-converter:3分钟实现B站缓存视频永久保存的终极方案 【免费下载链接】m4s-converter 一个跨平台小工具,将bilibili缓存的m4s格式音视频文件合并成mp4 项目地址: https://gitcode.com/gh_mirrors/m4/m4s-converter 你是否曾经在B站缓存了珍贵的…...