YYYY-MM-DD 的黑鍋,我們不背!

  • 2020 年 2 月 17 日
  • 筆記

點擊上方「挨踢小子」,「選擇關注」OR「設置星標」

執著派,技術流

作者:兔子托尼啊

https://zhuanlan.zhihu.com/p/101150248

寫這篇博文是記錄下跨年的bug。

去年隔壁組的小夥伴就是計算兩個日期之間間隔的天數,因為跨年的原因計算有誤。

當時測試組的小姐姐也沒有模擬出來這種場景,導致上生產環境直接影響線上的數據。

今天逛技術論論壇正好遇到Java日期的操作bug。

1 yyyy 和 YYYY

別看字,看程式碼

@Test  public 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" -> %sn",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也是不一樣要切記。

此鍋我們不背。

END

推薦一位有走心的coder,致力於打造一款高品質技術流學習社群,他專註於分享Java技術乾貨,包括面試攻略,開發技巧,架構設計,職場心得等。