Back to Tobetopjavaer

YYYY Vs Yyyy

docs/basics/java-basic/YYYY-vs-yyyy.md

latest3.0 KB
Original Source

在使用SimpleDateFormat的时候,需要通过字母来描述时间元素,并组装成想要的日期和时间模式。常用的时间元素和字母的对应表(JDK 1.8)如下:

可以看到,y表示Year ,而Y表示Week Year

什么是Week Year

我们知道,不同的国家对于一周的开始和结束的定义是不同的。如在中国,我们把星期一作为一周的第一天,而在美国,他们把星期日作为一周的第一天。

同样,如何定义哪一周是一年当中的第一周?这也是一个问题,有很多种方式。

比如下图是2019年12月-2020年1月的一份日历。

到底哪一周才算2020年的第一周呢?不同的地区和国家,甚至不同的人,都有不同的理解。

  • 1、1月1日是周三,到下周三(1月8日),这7天算作这一年的第一周。
  • 2、因为周日(周一)才是一周的第一天,所以,要从2020年的第一个周日(周一)开始往后推7天才算这一年的第一周。
  • 3、因为12.29、12.30、12.31是2019年,而1.1、1.2、1.3才是2020年,而1.4周日是下一周的开始,所以,第一周应该只有1.1、1.2、1.3这三天。

ISO 8601

因为不同人对于日期和时间的表示方法有不同的理解,于是,大家就共同制定了了一个国际规范:ISO 8601 。

国际标准化组织的国际标准ISO 8601是日期和时间的表示方法,全称为《数据存储和交换形式·信息交换·日期和时间的表示方法》。

在 ISO 8601中。对于一年的第一个日历星期有以下四种等效说法:

  • 1,本年度第一个星期四所在的星期;
  • 2,1月4日所在的星期;
  • 3,本年度第一个至少有4天在同一星期内的星期;
  • 4,星期一在去年12月29日至今年1月4日以内的星期;

根据这个标准,我们可以推算出:

2020年第一周:2019.12.29-2020.1.4

所以,根据ISO 8601标准,2019年12月29日、2019年12月30日、2019年12月31日这两天,其实不属于2019年的最后一周,而是属于2020年的第一周。

JDK针对ISO 8601提供的支持

根据ISO 8601中关于日历星期和日表示法的定义,2019.12.29-2020.1.4是2020年的第一周。

我们希望输入一个日期,然后程序告诉我们,根据ISO 8601中关于日历日期的定义,这个日期到底属于哪一年。

比如我输入2019-12-20,他告诉我是2019;而我输入2019-12-30的时候,他告诉我是2020。

为了提供这样的数据,Java 7引入了「YYYY」作为一个新的日期模式来作为标识。使用「YYYY」作为标识,。再通过SimpleDateFormat就可以得到一个日期所属的周属于哪一年了

所以,当我们要表示日期的时候,一定要使用 yyyy-MM-dd 而不是 YYYY-MM-dd ,这两者的返回结果大多数情况下都一样,但是极端情况就会有问题了。