SimpleDateFormat(“YYYY-MM-dd“)格式化时间出现了bug?

近期跨年的时候遇到了这样的时间格式化问题:

java 复制代码
Calendar ca = Calendar.getInstance();
ca.set(Calendar.YEAR, 2025);
ca.set(Calendar.MONTH, Calendar.DECEMBER);
ca.set(Calendar.DAY_OF_MONTH, 25);

SimpleDateFormat sdf = new SimpleDateFormat("YYYY-MM-dd");
//输出:2025-12-25,没问题
System.out.println(sdf.format(ca.getTime()));

ca.set(Calendar.DAY_OF_MONTH, 30);
//输出:2026-12-30,有问题,我的预期是输出:2025-12-30
System.out.println(sdf.format(ca.getTime()));

这导致了什么问题?

  • 我们系统的某个业务功能要按照年月日来生成文件夹和命名文件,由于使用了这种时间的格式化,导致了系统奇怪地出现了26年12月的文件夹,明明是在2025年啊;有些文件也被命名为了"20261231xxxx"。
  • 年末3天的缓存数据,放入是成功的,但是获取就是获取不到,key奇怪地出现了20261231,明明期望的应该是20251231。
  • 热点讨论功能前端要展示时间:刚刚、5分钟前、昨天 14:30、3天前、一年前。2025年12月29日这天的数据,前端居然永远展示"刚刚"。

问题点在哪里?

  • 格式化年份的占位符使用了YYYY。
  • 在SimpleDateFormat中,YYYY 是基于周(Week-Based-Year)的年份,而不是基于日期所在年的年份。

++什么是 Week-Based-Year?++

ISO-8601 标准:

  • 一周从星期一开始
  • 一年中的第一周,必须包含至少 4 天的新年

这就导致一个现象:有些 1 月初的日期,可能仍属于"上一年的最后一周"。

比如2025年12月的29、30、31日这三天,如果是Week-Based-Year,那么则归属是2026年,而不是2025年!

观察日历, 得出一个简单的结论: 新的一年所在的那一周,含有上一年的日期, YYYY方式 格式化 都会格式化错误。

**什么时候才应该使用 YYYY?**只有在做"基于周的统计"时才使用:

  • 周报系统
  • 财务周统计
  • 周序号计算

++怎么解决?++

**所以解决方案是:**时间格式化时用yyyy-MM-dd✅,而不是YYYY-MM-dd❌。

java 复制代码
@Test
public void test() {
    Calendar ca = Calendar.getInstance();
    ca.set(Calendar.YEAR, 2025);
    ca.set(Calendar.MONTH, Calendar.DECEMBER);
    ca.set(Calendar.DAY_OF_MONTH, 25);

    SimpleDateFormat sdf_wrong = new SimpleDateFormat("YYYY-MM-dd");
    SimpleDateFormat sdf_right = new SimpleDateFormat("yyyy-MM-dd");

    //输出:2025-12-25,没问题
    System.out.println(sdf_wrong.format(ca.getTime()));
    System.out.println(sdf_right.format(ca.getTime()));

    ca.set(Calendar.DAY_OF_MONTH, 30);
    System.out.println(sdf_wrong.format(ca.getTime())); //输出:2026-12-30,有问题,我的预期是输出:2025-12-30
    System.out.println(sdf_right.format(ca.getTime()));//输出:2025-12-30,没问题

    String format1 = DateTimeFormatter.ofPattern("YYYY-MM-dd").format(LocalDate.of(2025, 12, 31));
    System.out.println(format1);//输出:2025-12-31,没问题
    String format2 = DateTimeFormatter.ofPattern("yyyy-MM-dd").format(LocalDate.of(2025, 12, 31));
    System.out.println(format2);//输出:2025-12-31,没问题
}

++结论++

  • 时间格式化年份时一定要用yyyy,而不是其他。
  • 建议立刻全局搜索YYYY替换为yyyy,除非你非常明确是基于周的年份。
  • 建议使用DateTimeFormatter,而不是SimpleDateFormat。

本文结束。

相关推荐
查古穆1 小时前
LLM的“小bug”:聊聊幻觉是什么,以及如何有效规避免
人工智能·bug
程序员 沐阳3 小时前
Git 二分法精准定位 Bug:从原理到实战,让调试效率起飞
git·elasticsearch·bug
软泡芙5 小时前
【Bug】ReactiveUI WPF绑定中依赖属性不更新的问题分析与解决方案
java·bug·wpf
Yiyi_Coding1 天前
BUG列表:如何定位线上 OOM ?
java·linux·bug
csdn_aspnet3 天前
Git二分法精准定位Bug,分享用git bisect快速锁定引入缺陷的提交,提升调试效率
git·bug·二分查找
R6bandito_3 天前
自实现FLASH读取函数中的隐式类型转换bug踩坑记录
c语言·开发语言·经验分享·stm32·单片机·mcu·bug
奶茶精Gaaa3 天前
精彩bug--连续接受消息快速点击聊天页出现消息重叠
bug
奶茶精Gaaa3 天前
精彩bug--带图片+文字消息打开图片显示格式损坏
bug
奶茶精Gaaa3 天前
精彩bug--每次发版页面不会自动清缓存
bug
qq_452396234 天前
【测试之道】第六篇:缺陷管理论 —— Bug 的生态位、生命周期与根因分析(RCA)
功能测试·bug·软件工程