YYYY-MM-DD 的黑锅,我们不背!

自闭症网瘾萝莉.ら 提交于 2020-11-22 01:07:07

作者:兔子托尼啊
https://zhuanlan.zhihu.com/p/101150248
写这篇博文是记录下跨年的bug。
去年隔壁组的小伙伴就是计算两个日期之间间隔的天数,因为跨年的原因计算有误。
当时测试组的小姐姐也没有模拟出来这种场景,导致上生产环境直接影响线上的数据。
今天逛技术论论坛正好遇到 Java 日期的操作bug。

1 yyyy 和 YYYY

别看字,看代码
@Testpublic void testWeekBasedYear() {  Calendar calendar = Calendar.getInstance();  // 2019-12-31  calendar.set(2019, Calendar.DECEMBER, 31);  Date strDate1 = calendar.getTime();  // 2020-01-01  calendar.set(2020, Calendar.JANUARY, 1);  Date strDate2 = calendar.getTime();  // 大写 YYYY  SimpleDateFormat formatYYYY = new SimpleDateFormat("YYYY/MM/dd");  System.out.println("2019-12-31 转 YYYY/MM/dd 格式: " + formatYYYY.format(strDate1));  System.out.println("2020-01-01 转 YYYY/MM/dd 格式: " + formatYYYY.format(strDate2));  // 小写 YYYY  SimpleDateFormat formatyyyy = new SimpleDateFormat("yyyy/MM/dd");  System.out.println("2019-12-31 转 yyyy/MM/dd 格式: " + formatyyyy.format(strDate1));  System.out.println("2020-01-01 转 yyyy/MM/dd 格式: " + formatyyyy.format(strDate2));}

输出结果:


   
   
2019-12-31 转 YYYY/MM/dd 格式: 2020/12/31
2020-01-01 转 YYYY/MM/dd 格式: 2020/01/01
2019-12-31 转 yyyy/MM/dd 格式: 2019/12/31
2020-01-01 转 yyyy/MM/dd 格式: 2020/01/01
细心的同学应该发现了2019-12-31YYYY/MM/dd 此刻变成了2020/12/31
??为何呢?
YYYY这么大的能耐,能跑到 2020 年代去?
我2019年底买的东西,你如果用YYYY来格式化出库日期,我是不是得到2020年底才能收到货?此bug问题挺大的呀!

YYYY 到底是何方妖怪?👺


Java's DateTimeFormatter pattern "YYYY" gives you the week-based-year, (by default, ISO-8601 standard) the year of the Thursday of that week.


例子:

下面就是用YYYY格式化代码
  • 12/29/2019 将会格式化到2019年 这一周还属于2019年
  • 12/30/2019 将会格式化到2020年 这一周已经属于2020年
看字说话YYYY,week-based year 是 ISO 8601 规定的。
2019-12-31号这一天,安周算年份已经属于2020年了,格式化之后就变成2020年,后面的月份日期不变。

2 dd 和 DD

   
     
   
   
private static void tryit(int Y, int M, int D, String pat) {  DateTimeFormatter fmt = DateTimeFormatter.ofPattern(pat);  LocalDate dat = LocalDate.of(Y,M,D);  String str = fmt.format(dat);  System.out.printf("Y=%04d M=%02d D=%02d " +    "formatted with " +    "\"%s\" -> %s\n",Y,M,D,pat,str);}public static void main(String[] args){  tryit(2020,01,20,"MM/DD/YYYY");  tryit(2020,01,21,"DD/MM/YYYY");  tryit(2020,01,22,"YYYY-MM-DD");  tryit(2020,03,17,"MM/DD/YYYY");  tryit(2020,03,18,"DD/MM/YYYY");  tryit(2020,03,19,"YYYY-MM-DD");}


输出结果:


   
   
Y=2020 M=01 D=20 formatted with "MM/DD/YYYY" -> 01/20/2020
Y=2020 M=01 D=21 formatted with "DD/MM/YYYY" -> 21/01/2020
Y=2020 M=01 D=22 formatted with "YYYY-MM-DD" -> 2020-01-22
Y=2020 M=03 D=17 formatted with "MM/DD/YYYY" -> 03/77/2020
Y=2020 M=03 D=18 formatted with "DD/MM/YYYY" -> 78/03/2020
Y=2020 M=03 D=19 formatted with "YYYY-MM-DD" -> 2020-03-79
看到没有?
最后的3个日期都错误了,这里的大写的DD代表的是处于这一年中那一天,不是处于这个月的那一天。
小伙伴们一定要记住了不要犯类似的错误。

3、结论

YYYY和yyyy不一样的,DD和dd也是不一样要切记。
此锅我们不背。
推荐阅读:
挑战10个最难回答的Java面试题
面试|JAVA 引用详解
Java 程序员必须了解的 7 个性能指标

本文分享自微信公众号 - 浪尖聊大数据(bigdatatip)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!