除了lambda表達(dá)式,stream以及幾個(gè)小的改進(jìn)之外,Java 8還引入了一套全新的時(shí)間日期API,在本篇教程中我們將通過幾個(gè)簡(jiǎn)單的任務(wù)示例來學(xué)習(xí)如何使用Java 8的這套API。Java對(duì)日期,日歷及時(shí)間的處理一直以來都飽受詬病,尤其是它決定將java.util.Date定義為可修改的以及將SimpleDateFormat實(shí)現(xiàn)成非線程安全的。看來Java已經(jīng)意識(shí)到需要為時(shí)間及日期功能提供更好的支持了,這對(duì)已經(jīng)習(xí)慣使用Joda時(shí)間日期庫的社區(qū)而言也是件好事。關(guān)于這個(gè)新的時(shí)間日期庫的最大的優(yōu)點(diǎn)就在于它定義清楚了時(shí)間日期相關(guān)的一些概念,比方說,瞬時(shí)時(shí)間(Instant),持續(xù)時(shí)間(duration),日期(date),時(shí)間(time),時(shí)區(qū)(time-zone)以及時(shí)間段(Period)。同時(shí)它也借鑒了Joda庫的一些優(yōu)點(diǎn),比如將人和機(jī)器對(duì)時(shí)間日期的理解區(qū)分開的。Java 8仍然延用了ISO的日歷體系,并且與它的前輩們不同,java.time包中的類是不可變且線程安全的。新的時(shí)間及日期API位于java.time包中,下面是里面的一些關(guān)鍵的類:
1.Instant――它代表的是時(shí)間戳
2.LocalDate――不包含具體時(shí)間的日期,比如2014-01-14。它可以用來存儲(chǔ)生日,周年紀(jì)念日,入職日期等。
3.LocalTime――它代表的是不含日期的時(shí)間
4.LocalDateTime――它包含了日期及時(shí)間,不過還是沒有偏移信息或者說時(shí)區(qū)。
5.ZonedDateTime――這是一個(gè)包含時(shí)區(qū)的完整的日期時(shí)間,偏移量是以UTC/格林威治時(shí)間為基準(zhǔn)的。
新的庫還增加了ZoneOffset及Zoned,可以為時(shí)區(qū)提供更好的支持。有了新的DateTimeFormatter之后日期的解析及格式化也變得煥然一新了。隨便提一句,我是在去年這個(gè)時(shí)候Java正要推出這個(gè)新功能時(shí)寫的這篇文章,所以你會(huì)發(fā)現(xiàn)示例中的時(shí)間都還是去年的。你運(yùn)行下這些例子,它們返回的值肯定都是正確的。
Java8是如何處理時(shí)間及日期的
有人問我學(xué)習(xí)一個(gè)新庫的最佳途徑是什么?我的回答是,就是在實(shí)際項(xiàng)目中那樣去使用它。在一個(gè)真實(shí)的項(xiàng)目中會(huì)有各種各樣的需求,這會(huì)促使開發(fā)人員去探索和研究這個(gè)新庫。簡(jiǎn)言之,只有任務(wù)本身才會(huì)真正促使你去探索及學(xué)習(xí)。java 8的新的日期及時(shí)間API也是一樣。為了學(xué)習(xí)Java 8的這個(gè)新庫,這里我創(chuàng)建了20個(gè)以任務(wù)為導(dǎo)向的例子。我們先從一個(gè)簡(jiǎn)單的任務(wù)開始,比如說如何用Java 8的時(shí)間日期庫來表示今天,接著再進(jìn)一步生成一個(gè)帶時(shí)間及時(shí)區(qū)的完整日期,然后再研究下如何完成一些更實(shí)際的任務(wù),比如說開發(fā)一個(gè)提醒類的應(yīng)用,來找出距離一些特定日期比如生日,周日紀(jì)念日,下一個(gè)帳單日,下一個(gè)溢價(jià)日或者信用卡過期時(shí)間還有多少天。
示例1 如何 在Java 8中獲取當(dāng)天的日期
Java 8中有一個(gè)叫LocalDate的類,它能用來表示今天的日期。這個(gè)類與java.util.Date略有不同,因?yàn)樗话掌冢瑳]有時(shí)間。因此,如果你只需要表示日期而不包含時(shí)間,就可以使用它。
Output
Today's Local date : 2014-01-14
你可以看到它創(chuàng)建了今天的日期卻不包含時(shí)間信息。它還將日期格式化完了再輸出出來,不像之前的Date類那樣,打印出來的數(shù)據(jù)都是未經(jīng)格式化的。
示例2 如何在Java 8中獲取當(dāng)前的年月日
LocalDate類中提供了一些很方便的方法可以用于提取出年月日以及其它的日期屬性。使用這些方法,你可以獲取到任何你所需要的日期屬性,而不再需要使用java.util.Calendar這樣的類了:
Output
Today's Local date : 2014-01-14
Year : 2014 Month : 1 day : 14
可以看到,在Java 8中獲取年月信息非常簡(jiǎn)單,只需使用對(duì)應(yīng)的getter方法就好了,無需記憶,非常直觀。你可以拿它和Java中老的獲取當(dāng)前年月日的寫法進(jìn)行一下比較。
示例3 在Java 8中如何獲取某個(gè)特定的日期
在第一個(gè)例子中,我們看到通過靜態(tài)方法now()來生成當(dāng)天日期是非常簡(jiǎn)單的,不過通過另一個(gè)十分有用的工廠方法LocalDate.of(),則可以創(chuàng)建出任意一個(gè)日期,它接受年月日的參數(shù),然后返回一個(gè)等價(jià)的LocalDate實(shí)例。關(guān)于這個(gè)方法還有一個(gè)好消息就是它沒有再犯之前API中的錯(cuò),比方說,年只能從1900年開始,月必須從0開始,等等。這里的日期你寫什么就是什么,比如說,下面這個(gè)例子中它代表的就是1月14日,沒有什么隱藏邏輯。
Output : Your Date of birth is : 2010-01-14
可以看出,創(chuàng)建出來的日期就是我們所寫的那樣,2014年1月14日。
示例4 在Java 8中如何檢查兩個(gè)日期是否相等
如果說起現(xiàn)實(shí)中實(shí)際的處理時(shí)間及日期的任務(wù),有一個(gè)常見的就是要檢查兩個(gè)日期是否相等。你可能經(jīng)常會(huì)碰到要判斷今天是不是某個(gè)特殊的日子,比如生日啊,周年紀(jì)念日啊,或者假期之類。有的時(shí)候,會(huì)給你一個(gè)日期,讓你檢查它是不是某個(gè)日子比方說假日。下面這個(gè)例子將會(huì)幫助你在Java 8中完成這類任務(wù)。正如你所想的那樣,LocalDate重寫了equals方法來進(jìn)行日期的比較,如下所示:
Output
today 2014-01-14 and date1 2014-01-14 are same date
在本例中我們比較的兩個(gè)日期是相等的。同時(shí),如果在代碼中你拿到了一個(gè)格式化好的日期串,你得先將它解析成日期然后才能比較。你可以將這個(gè)例子與Java之前比較日期的方式進(jìn)行下比較,你會(huì)發(fā)現(xiàn)它真是爽多了。
示例5 在Java 8中如何檢查重復(fù)事件,比如說生日
在Java中還有一個(gè)與時(shí)間日期相關(guān)的實(shí)際任務(wù)就是檢查重復(fù)事件,比如說每月的帳單日,結(jié)婚紀(jì)念日,每月還款日或者是每年交保險(xiǎn)費(fèi)的日子。如果你在一家電商公司工作的話,那么肯定會(huì)有這么一個(gè)模塊,會(huì)去給用戶發(fā)送生日祝福并且在每一個(gè)重要的假日給他們捎去問候,比如說圣誕節(jié),感恩節(jié),在印度則可能是萬燈節(jié)(Deepawali)。如何在Java中判斷是否是某個(gè)節(jié)日或者重復(fù)事件?使用MonthDay類。這個(gè)類由月日組合,不包含年信息,也就是說你可以用它來代表每年重復(fù)出現(xiàn)的一些日子。當(dāng)然也有一些別的組合,比如說YearMonth類。它和新的時(shí)間日期庫中的其它類一樣也都是不可變且線程安全的,并且它還是一個(gè)值類(value class)。我們通過一個(gè)例子來看下如何使用MonthDay來檢查某個(gè)重復(fù)的日期:
Output: Many Many happy returns of the day !!
示例6 如何在Java 8中獲取當(dāng)前時(shí)間
這與第一個(gè)例子中獲取當(dāng)前日期非常相似。這次我們用的是一個(gè)叫LocalTime的類,它是沒有日期的時(shí)間,與LocalDate是近親。這里你也可以用靜態(tài)工廠方法now()來獲取當(dāng)前時(shí)間。默認(rèn)的格式是hh:mm:ss:nnn,這里的nnn是納秒。可以和Java 8以前如何獲取當(dāng)前時(shí)間做一下比較。
Output
local time now : 16:33:33.369 // in hour, minutes, seconds, nano seconds
示例7 如何增加時(shí)間里面的小時(shí)數(shù)
很多時(shí)候我們需要增加小時(shí),分或者秒來計(jì)算出將來的時(shí)間。Java 8不僅提供了不可變且線程安全的類,它還提供了一些更方便的方法譬如plusHours()來替換原來的add()方法。順便說一下,這些方法返回的是一個(gè)新的LocalTime實(shí)例的引用,因?yàn)長(zhǎng)ocalTime是不可變的,可別忘了存儲(chǔ)好這個(gè)新的引用。
Output :
Time after 2 hours : 18:33:33.369
示例8 如何獲取1周后的日期
這與前一個(gè)獲取2小時(shí)后的時(shí)間的例子類似,這里我們將學(xué)會(huì)如何獲取到1周后的日期。LocalDate是用來表示無時(shí)間的日期的,它有一個(gè)plus()方法可以用來增加日,星期,或者月,ChronoUnit則用來表示這個(gè)時(shí)間單位。由于LocalDate也是不可變的,因此任何修改操作都會(huì)返回一個(gè)新的實(shí)例,因此別忘了保存起來。
Output:
Today is : 2014-01-14
Date after 1 week : 2014-01-21
示例9 一年前后的日期
這是上個(gè)例子的續(xù)集。上例中,我們學(xué)習(xí)了如何使用LocalDate的plus()方法來給日期增加日,周或者月,現(xiàn)在我們來學(xué)習(xí)下如何用minus()方法來找出一年前的那天。
Output:
Date before 1 year : 2013-01-14
Date after 1 year : 2015-01-14
示例10 在Java 8中使用時(shí)鐘
Java 8中自帶了一個(gè)Clock類,你可以用它來獲取某個(gè)時(shí)區(qū)下當(dāng)前的瞬時(shí)時(shí)間,日期或者時(shí)間。可以用Clock來替代System.currentTimeInMillis()與 TimeZone.getDefault()方法。
// Returns time based on system clock zone Clock defaultClock =
Clock.systemDefaultZone();
System.out.println("Clock : " + clock);
Output:
Clock : SystemClock[Z]
Clock : SystemClock[Z]
public void process(LocalDate eventDate) {
if(eventDate.isBefore(LocalDate.now(clock)) {
...
}
}
}
示例11 在Java中如何判斷某個(gè)日期是在另一個(gè)日期的前面還是后面
這也是實(shí)際項(xiàng)目中常見的一個(gè)任務(wù)。你怎么判斷某個(gè)日期是在另一個(gè)日期的前面還是后面,或者正好相等呢?在Java 8中,LocalDate類有一個(gè)isBefore()和isAfter()方法可以用來比較兩個(gè)日期。如果調(diào)用方法的那個(gè)日期比給定的日期要早的話,isBefore()方法會(huì)返回true。
Output:
Tomorrow comes after today
Yesterday is day before today
示例12 在Java 8中處理不同的時(shí)區(qū)
Java 8不僅將日期和時(shí)間進(jìn)行了分離,同時(shí)還有時(shí)區(qū)。現(xiàn)在已經(jīng)有好幾組與時(shí)區(qū)相關(guān)的類了,比如ZonId代表的是某個(gè)特定的時(shí)區(qū),而ZonedDateTime代表的是帶時(shí)區(qū)的時(shí)間。它等同于Java 8以前的GregorianCalendar類。使用這個(gè)類,你可以將本地時(shí)間轉(zhuǎn)換成另一個(gè)時(shí)區(qū)中的對(duì)應(yīng)時(shí)間,比如下面這個(gè)例子:
Output :
Current date and time in a particular timezone : 2014-01-14T16:33:33.373-05:00[America/New_York]
可以拿它跟之前將本地時(shí)間轉(zhuǎn)換成GMT時(shí)間的方式進(jìn)行下比較。順便說一下,正如Java 8以前那樣,對(duì)應(yīng)時(shí)區(qū)的那個(gè)文本可別弄錯(cuò)了,否則你會(huì)碰到這么一個(gè)異常:
示例13 如何表示固定的日期,比如信用卡過期時(shí)間
正如MonthDay表示的是某個(gè)重復(fù)出現(xiàn)的日子的,YearMonth又是另一個(gè)組合,它代表的是像信用卡還款日,定期存款到期日,options到期日這類的日期。你可以用這個(gè)類來找出那個(gè)月有多少天,lengthOfMonth()這個(gè)方法返回的是這個(gè)YearMonth實(shí)例有多少天,這對(duì)于檢查2月到底是28天還是29天可是非常有用的。
Output:
Days in month year 2014-01: 31
Your credit card expires on 2018-02
示例14 如何在Java 8中檢查閏年
這并沒什么復(fù)雜的,LocalDate類有一個(gè)isLeapYear()的方法能夠返回當(dāng)前LocalDate對(duì)應(yīng)的那年是否是閏年。如果你還想重復(fù)造輪子的話,可以看下這段代碼,這是純用Java編寫的判斷某年是否是閏年的邏輯。
Output: 2014 is not a Leap year
示例15 兩個(gè)日期之間包含多少天,多少個(gè)月
還有一個(gè)常見的任務(wù)就是計(jì)算兩個(gè)給定的日期之間包含多少天,多少周或者多少年。你可以用java.time.Period類來完成這個(gè)功能。在下面這個(gè)例子中,我們將計(jì)算當(dāng)前日期與將來的一個(gè)日期之前一共隔著幾個(gè)月。
Output:
Months left between today and Java 8 release : 2
示例16 帶時(shí)區(qū)偏移量的日期與時(shí)間
在Java 8里面,你可以用ZoneOffset類來代表某個(gè)時(shí)區(qū),比如印度是GMT或者UTC5:30,你可以使用它的靜態(tài)方法ZoneOffset.of()方法來獲取對(duì)應(yīng)的時(shí)區(qū)。只要獲取到了這個(gè)偏移量,你就可以拿LocalDateTime和這個(gè)偏移量創(chuàng)建出一個(gè)OffsetDateTime。
Output :
Date and Time with timezone offset in Java : 2014-01-14T19:30+05:30
示例17 在Java 8中如何獲取當(dāng)前時(shí)間戳
如果你還記得在Java 8前是如何獲取當(dāng)前時(shí)間戳的,那現(xiàn)在這簡(jiǎn)直就是小菜一碟了。Instant類有一個(gè)靜態(tài)的工廠方法now()可以返回當(dāng)前時(shí)間戳,如下:
Output :
What is value of this instant 2014-01-14T08:33:33.379Z
示例18 如何在Java 8中使用預(yù)定義的格式器來對(duì)日期進(jìn)行解析/格式化
在Java 8之前,時(shí)間日期的格式化可是個(gè)技術(shù)活,我們的好伙伴SimpleDateFormat并不是線程安全的,而如果用作本地變量來格式化的話又顯得有些笨重。多虧了線程本地變量,這使得它在多線程環(huán)境下也算有了用武之地,但Java維持這一狀態(tài)也有很長(zhǎng)一段時(shí)間了。這次它引入了一個(gè)全新的線程安全的日期與時(shí)間格式器。它還自帶了一些預(yù)定義好的格式器,包含了常用的日期格式。比如說,本例 中我們就用了預(yù)定義的BASICISODATE格式,它會(huì)將2014年2月14日格式化成20140114。
Output :
Date generated from String 20140116 is 2014-01-16
示例19 如何在Java中使用自定義的格式器來解析日期
在上例中,我們使用了內(nèi)建的時(shí)間日期格式器來解析日期字符串。當(dāng)然了,預(yù)定義的格式器的確不錯(cuò)但有時(shí)候你可能還是需要使用自定義的日期格式,這個(gè)時(shí)候你就得自己去創(chuàng)建一個(gè)自定義的日期格式器實(shí)例了。下面這個(gè)例子中的日期格式是”MMM dd yyyy”。你可以給DateTimeFormatter的ofPattern靜態(tài)方法()傳入任何的模式,它會(huì)返回一個(gè)實(shí)例,這個(gè)模式的字面量與前例中是相同的。比如說M還是代表月,而m仍是分。無效的模式會(huì)拋出DateTimeParseException異常,但如果是邏輯上的錯(cuò)誤比如說該用M的時(shí)候用成m,這樣就沒辦法了。
Output :
Successfully parsed String Apr 18 2014, date is 2014-04-18
示例20 如何在Java 8中對(duì)日期進(jìn)行格式化,轉(zhuǎn)換成字符串
在上兩個(gè)例子中,盡管我們用到了DateTimeFormatter類但我們主要是進(jìn)行日期字符串的解析。在這個(gè)例子中我們要做的事情正好相反。這里我們有一個(gè)LocalDateTime類的實(shí)例,我們要將它轉(zhuǎn)換成一個(gè)格式化好的日期串。這是目前為止Java中將日期轉(zhuǎn)換成字符串最簡(jiǎn)單便捷的方式了。下面這個(gè)例子將會(huì)返回一個(gè)格式化好的字符串。與前例相同的是,我們?nèi)孕枋褂弥付ǖ哪J酱?chuàng)建一個(gè)DateTimeFormatter類的實(shí)例,但調(diào)用的并不是LocalDate類的parse方法,而是它的format()方法。這個(gè)方法會(huì)返回一個(gè)代表當(dāng)前日期的字符串,對(duì)應(yīng)的模式就是傳入的DateTimeFormatter實(shí)例中所定義好的。
Output : Arriving at : Jan 14 2014 04:33 PM
Java 8中日期與時(shí)間API的幾個(gè)關(guān)鍵點(diǎn)
看完了這些例子后,我相信你已經(jīng)對(duì)Java 8這套新的時(shí)間日期API有了一定的了解了。現(xiàn)在我們來回顧下關(guān)于這個(gè)新的API的一些關(guān)鍵的要素。
1.它提供了javax.time.ZoneId用來處理時(shí)區(qū)。
2.它提供了LocalDate與LocalTime類
3.Java 8中新的時(shí)間與日期API中的所有類都是不可變且線程安全的,這與之前的Date與Calendar API中的恰好相反,那里面像java.util.Date以及SimpleDateFormat這些關(guān)鍵的類都不是線程安全的。
4.新的時(shí)間與日期API中很重要的一點(diǎn)是它定義清楚了基本的時(shí)間與日期的概念,比方說,瞬時(shí)時(shí)間,持續(xù)時(shí)間,日期,時(shí)間,時(shí)區(qū)以及時(shí)間段。它們都是基于ISO日歷體系的。
5.每個(gè)Java開發(fā)人員都應(yīng)該至少了解這套新的API中的這五個(gè)類:
5.1)Instant 它代表的是時(shí)間戳,比如2014-01-14T02:20:13.592Z,這可以從java.time.Clock類中獲取,像這樣: Instant current = Clock.system(ZoneId.of(“Asia/Tokyo”)).instant();
5.2)LocalDate 它表示的是不帶時(shí)間的日期,比如2014-01-14。它可以用來存儲(chǔ)生日,周年紀(jì)念日,入職日期等。
5.3)LocalTime
主站蜘蛛池模板:
江北区|
长兴县|
肥乡县|
宁国市|
买车|
忻城县|
南昌市|
普安县|
浦城县|
荆州市|
建始县|
海丰县|
苍山县|
休宁县|
漾濞|
黔西县|
临澧县|
柘荣县|
临朐县|
曲阳县|
和平县|
琼海市|
图木舒克市|
林西县|
资中县|
陆良县|
罗平县|
洞口县|
瑞金市|
柞水县|
始兴县|
天门市|
平乐县|
南康市|
定结县|
赞皇县|
宜春市|
彩票|
乌鲁木齐县|
兴安县|
宜昌市|