為什么阿里巴巴要求日期格式化時(shí)必須有使用y表示年
在Java中進(jìn)行日期處理大家一定都不陌生,我們經(jīng)常會(huì)需要在代碼中進(jìn)行日期的轉(zhuǎn)換、日期的格式化等操作。
而一般我們進(jìn)行日期格式化的時(shí)候都會(huì)使用SimpleDateFormat工具,之前我們有一篇文章介紹過SimpleDateFormat的線程安全問題(https://www.hollischuang.com/archives/3017),這一篇文章再來介紹一個(gè)和SimpleDateFormat有關(guān),很容易被忽視,而一旦忽視可能導(dǎo)致大故障的問題。
SimpleDateFormat
SimpleDateFormat是Java提供的一個(gè)格式化和解析日期的工具類。它允許進(jìn)行格式化(日期 -> 文本)、解析(文本 -> 日期)和規(guī)范化。SimpleDateFormat 使得可以選擇任何用戶定義的日期-時(shí)間格式的模式。
在Java中,可以使用SimpleDateFormat的format方法,將一個(gè)Date類型轉(zhuǎn)化成String類型,并且可以指定輸出格式。
// Date轉(zhuǎn)String Date data = new Date(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dataStr = sdf.format(data); System.out.println(dataStr);
以上代碼,轉(zhuǎn)換的結(jié)果是:2018-11-25 13:00:00,日期和時(shí)間格式由”日期和時(shí)間模式”字符串指定。如果你想要轉(zhuǎn)換成其他格式,只要指定不同的時(shí)間模式就行了。
在Java中,可以使用SimpleDateFormat的parse方法,將一個(gè)String類型轉(zhuǎn)化成Date類型。
// String轉(zhuǎn)Data System.out.println(sdf.parse(dataStr));
日期和時(shí)間模式表達(dá)方法
在使用SimpleDateFormat的時(shí)候,需要通過字母來描述時(shí)間元素,并組裝成想要的日期和時(shí)間模式。常用的時(shí)間元素和字母的對(duì)應(yīng)表(JDK 1.8)如下:
模式字母通常是重復(fù)的,其數(shù)量確定其精確表示。如前面我們用過的"yyyy-MM-dd HH:mm:ss"。我們知道其中的y其實(shí)是year的縮寫,所以我們都知道使用y來表示年,默認(rèn)情況下,我們都會(huì)使用y而不是Y,那么這兩者之間有何區(qū)別呢?一旦用錯(cuò)了會(huì)帶來什么后果呢?
其實(shí)在規(guī)定中,y表示year,而Y表示W(wǎng)eek Year!
什么是Week Year
我們知道,不同的國家對(duì)于一周的開始和結(jié)束的定義是不同的。如在中國,我們把星期一作為一周的第一天,而在美國,他們把星期日作為一周的第一天。
同樣,如何定義哪一周是一年當(dāng)中的第一周?這也是一個(gè)問題,有很多種方式。
比如下圖是2019年12月-2020年1月的一份日歷。
到底哪一周才算2020年的第一周呢?不同的地區(qū)和國家,甚至不同的人,都有不同的理解。
1、1月1日是周三,到下周三(1月8日),這7天算作這一年的第一周。
2、因?yàn)橹苋眨ㄖ芤唬┎攀且恢艿牡谝惶欤?,要?020年的第一個(gè)周日(周一)開始往后推7天才算這一年的第一周。
3、因?yàn)?2.29、12.30、12.31是2019年,而1.1、1.2、1.3才是2020年,而1.4周日是下一周的開始,所以,第一周應(yīng)該只有1.1、1.2、1.3這三天。
ISO 8601
因?yàn)椴煌藢?duì)于日期和時(shí)間的表示方法有不同的理解,于是,大家就共同制定了了一個(gè)國際規(guī)范:ISO 8601 。
國際標(biāo)準(zhǔn)化組織的國際標(biāo)準(zhǔn)ISO 8601是日期和時(shí)間的表示方法,全稱為《數(shù)據(jù)存儲(chǔ)和交換形式·信息交換·日期和時(shí)間的表示方法》。
在 ISO 8601中。對(duì)于一年的第一個(gè)日歷星期有以下四種等效說法:1,本年度第一個(gè)星期四所在的星期;2,1月4日所在的星期;3,本年度第一個(gè)至少有4天在同一星期內(nèi)的星期;4,星期一在去年12月29日至今年1月4日以內(nèi)的星期;
根據(jù)這個(gè)標(biāo)準(zhǔn),我們可以推算出:
2020年第一周:2019.12.29-2020.1.4
所以,根據(jù)ISO 8601標(biāo)準(zhǔn),2019年12月29日、2019年12月30日、2019年12月31日這兩天,其實(shí)不屬于2019年的最后一周,而是屬于2020年的第一周。
JDK針對(duì)ISO 8601提供的支持
根據(jù)ISO 8601中關(guān)于日歷星期和日表示法的定義,2019.12.29-2020.1.4是2020年的第一周。
我們希望輸入一個(gè)日期,然后程序告訴我們,根據(jù)ISO 8601中關(guān)于日歷日期的定義,這個(gè)日期到底屬于哪一年。
比如我輸入2019-12-20,他告訴我是2019;而我輸入2019-12-30的時(shí)候,他告訴我是2020。
為了提供這樣的數(shù)據(jù),Java 7引入了「YYYY」作為一個(gè)新的日期模式來作為標(biāo)識(shí)。使用「YYYY」作為標(biāo)識(shí),再通過SimpleDateFormat就可以得到一個(gè)日期所屬的周屬于哪一年了。
所以,我們通過代碼可以驗(yàn)證:
public class WeekYearTest { public static void main(String[] args) { SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); SimpleDateFormat sdf1 = new SimpleDateFormat("YYYY"); System.out.println(sdf1.format(sdf.parse("2019-12-01"))); System.out.println(sdf1.format(sdf.parse("2019-12-30"))); System.out.println(sdf1.format(sdf.parse("2020-01-01"))); } }
輸出結(jié)果為:
2019
2020
2020
可見, 2019-12-30日因?yàn)閷儆?020年的第一周,所以返回的年份是2020年。
而如果將「YYYY」改成「yyyy」的話,輸出結(jié)果就為:
2019
2019
2020
因?yàn)橛羞@樣的情況,所以我們?nèi)粘i_發(fā)的時(shí)候,如果把y寫成了Y,那就可能導(dǎo)致日期輸出的結(jié)果不符合我們的預(yù)期。
當(dāng)我們要表示日期的時(shí)候,一定要使用 yyyy-MM-dd 而不是 YYYY-MM-dd ,這兩者的返回結(jié)果大多數(shù)情況下都一樣,但是極端情況就會(huì)有問題了。
而這一個(gè)要求,在《阿里巴巴Java開發(fā)手冊(cè)》中也有類似的規(guī)定:
好啦,大家快去排查下你的代碼,有沒有'YYYY-MM-dd'這種形式的代碼吧,如果有的話,一定要改掉哦!~
到此這篇關(guān)于為什么阿里巴巴要求日期格式化時(shí)必須有使用y表示年的文章就介紹到這了,更多相關(guān)阿里巴巴日期格式化y表示年內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java實(shí)現(xiàn)JDK動(dòng)態(tài)代理的原理詳解
這篇文章主要介紹了Java實(shí)現(xiàn)JDK動(dòng)態(tài)代理的原理詳解,Java常用的動(dòng)態(tài)代理模式有JDK動(dòng)態(tài)代理,也有cglib動(dòng)態(tài)代理,本文重點(diǎn)講解JDK的動(dòng)態(tài)代理,需要的小伙伴可以參考一下的相關(guān)資料2022-07-07Springboot web項(xiàng)目打包實(shí)現(xiàn)過程解析
這篇文章主要介紹了Springboot web項(xiàng)目打包實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08深入淺析Random類在高并發(fā)下的缺陷及JUC對(duì)其的優(yōu)化
這篇文章主要介紹了Random類在高并發(fā)下的缺陷及JUC對(duì)其的優(yōu)化 ,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-04-04在lambda的foreach遍歷中break退出操作(lambda foreach break)
這篇文章主要介紹了在lambda的foreach遍歷中break退出操作(lambda foreach break),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-09-09解決springboot3:mybatis-plus依賴錯(cuò)誤:org.springframework.beans.fac
這篇文章主要介紹了解決springboot3:mybatis-plus依賴錯(cuò)誤:org.springframework.beans.factory.UnsatisfiedDependencyException問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-07-07SpringBoot啟動(dòng)時(shí)自動(dòng)執(zhí)行代碼的幾種實(shí)現(xiàn)方式
這篇文章主要給大家介紹了關(guān)于SpringBoot啟動(dòng)時(shí)自動(dòng)執(zhí)行代碼的幾種實(shí)現(xiàn)方式,文中通過實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-02-02Java 覆蓋equals時(shí)總要覆蓋hashcode
這篇文章主要介紹了Java 覆蓋equals時(shí)總要覆蓋hashcode的相關(guān)資料,這里附有實(shí)例代碼,具有參考價(jià)值,需要的朋友可以參考下2016-12-12