Java?8?Time?Api?使用方法技巧
1.概述
作為本文的一部分,讓我們從現(xiàn)有Date和Calendar API存在的一些問題入手,來探討新的Java 8 Date和Time API如何解決這些問題。
我們還將搞一搞Java 8時(shí)間類庫中的核心類,比如LocalDate, LocalTime, LocalDateTime, ZonedDateTime, Period, Duration以及它們的api。
2. 舊的時(shí)間API(java8之前)的問題
- 線程安全 - Date 和Calendar類不是線程安全的,使開發(fā)者難以調(diào)試這些api的并發(fā)問題,需要編寫額外的代碼來處理線程安全。Java 8中引入的新的Date和Time API是不可變的和線程安全的,使得這些痛點(diǎn)得以解決。
- API設(shè)計(jì)和易于理解 - 舊的時(shí)間api非常難以理解,操作都非常復(fù)雜,非常繞口,沒有提供一些常用的解析轉(zhuǎn)換方法。新的時(shí)間API是以ISO為中心的,并遵循 date, time, duration 和 periods的一致域模型。提供了一些非常實(shí)用方法以支持最常見的操作。不再需要我們自己封裝一些時(shí)間操作類。
- ZonedDate和Time - 在舊的時(shí)間api中開發(fā)人員必須編寫額外的邏輯來處理舊API的時(shí)區(qū)邏輯,而使用新的API,可以使用 Local和ZonedDate / Time API來處理時(shí)區(qū)。無需過多關(guān)心時(shí)區(qū)轉(zhuǎn)換問題。
3.使用LocalDate,LocalTime和LocalDateTime
最常用的類是LocalDate,LocalTime和LocalDateTime。正如他們的名字所示,它們代表與上下文相結(jié)合的本地日期/時(shí)間。
這些類主要用于不需要在上下文中明確指定時(shí)區(qū)的情況。作為本節(jié)的一部分,我們將介紹最常用的API。
3.1.使用LocalDate
LocalDate表示在ISO格式(YYYY-MM-DD)下的不帶具體時(shí)間的日期。
常用于表示生日或者我們最關(guān)心的發(fā)工資的日期。
獲取當(dāng)前系統(tǒng)時(shí)鐘下的日期,如下所示:
LocalDate localDate = LocalDate.now();
表示特定日,月和年的LocalDate可以使用“ of ”方法或使用“ parse ”方法獲得。例如,以下代碼段代表2015年2月20日的LocalDate:
LocalDate.of(2015, 02, 20); LocalDate.parse("2015-02-20");
是不是非常直觀而且方便呢!LocalDate提供各種實(shí)用方法,以獲得各種日期信息。讓我們快速瀏覽一下這些API方法。
以下代碼段獲取當(dāng)前本地日期并添加一天:
LocalDate tomorrow = LocalDate.now().plusDays(1);
此示例獲取當(dāng)前日期并減去一個(gè)月。請(qǐng)注意它是如何接受枚舉作為時(shí)間單位的:
LocalDate previousMonthSameDay = LocalDate.now().minus(1, ChronoUnit.MONTHS);
在以下兩個(gè)代碼示例中,我們分析日期“2016-06-12”并分別獲取星期幾和月中的某天。注意返回值,第一個(gè)是表示DayOfWeek的對(duì)象,而第二個(gè)是表示月份的序數(shù)值的int:
DayOfWeek sunday = LocalDate.parse("2016-06-12").getDayOfWeek(); int twelve = LocalDate.parse("2016-06-12").getDayOfMonth();
我們可以測試一個(gè)日期是否發(fā)生在閏年,如果用老方法怕不是要上天:
boolean leapYear = LocalDate.now().isLeapYear();
判斷日期的先后:
boolean notBefore = LocalDate.parse("2016-06-12").isBefore(LocalDate.parse("2016-06-11")); boolean isAfter = LocalDate.parse("2016-06-12").isAfter(LocalDate.parse("2016-06-11"));
日期邊界可以從給定日期獲得。在以下兩個(gè)示例中,我們得到LocalDateTime,它代表給定日期的一天的開始(2016-06-12T00:00)和代表月初的LocalDate(2016-06-01):
LocalDateTime beginningOfDay = LocalDate.parse("2016-06-12").atStartOfDay(); LocalDate firstDayOfMonth = LocalDate.parse("2016-06-12") .with(TemporalAdjusters.firstDayOfMonth());
現(xiàn)在讓我們來看看我們?nèi)绾问褂卯?dāng)?shù)貢r(shí)間。
3.2.使用LocalTime
在本地時(shí)間表示不帶日期的時(shí)間。
與LocalDate類似,可以從系統(tǒng)時(shí)鐘或使用“parse”和“of”方法創(chuàng)建LocalTime實(shí)例??焖贋g覽下面的一些常用API。
可以從系統(tǒng)時(shí)鐘創(chuàng)建當(dāng)前LocalTime的實(shí)例,如下所示:
LocalTime now = LocalTime.now();
在下面的代碼示例中,我們通過解析字符串表示創(chuàng)建表示06:30 AM 的LocalTime:
LocalTime sixThirty = LocalTime.parse("06:30");
方法“of”可用于創(chuàng)建LocalTime。例如,下面的代碼使用“of”方法創(chuàng)建表示06:30 AM的LocalTime:
LocalTime sixThirty = LocalTime.of(6, 30);
下面的示例通過解析字符串來創(chuàng)建LocalTime,并使用“plus”API為其添加一小時(shí)。結(jié)果將是代表07:30 AM的LocalTime:
LocalTime sevenThirty = LocalTime.parse("06:30").plus(1, ChronoUnit.HOURS);
各種getter方法可用于獲取特定的時(shí)間單位,如小時(shí),分鐘和秒,如下所示獲取小時(shí):
int six = LocalTime.parse("06:30").getHour();
同LocalDate一樣檢查特定時(shí)間是否在另一特定時(shí)間之前或之后。下面的代碼示例比較結(jié)果為true的兩個(gè)LocalTime:
boolean isbefore = LocalTime.parse("06:30").isBefore(LocalTime.parse("07:30"));
一天中的最大,最小和中午時(shí)間可以通過LocalTime類中的常量獲得。在執(zhí)行數(shù)據(jù)庫查詢以查找給定時(shí)間范圍內(nèi)的記錄時(shí),這非常有用。例如,下面的代碼代表23:59:59.99:
LocalTime maxTime = LocalTime.MAX
現(xiàn)在讓我們深入了解LocalDateTime。
3.3.使用LocalDateTime
所述LocalDateTime用于表示日期和時(shí)間的組合。
當(dāng)我們需要結(jié)合日期和時(shí)間時(shí),這是最常用的類。該類提供了各種API,我們將介紹一些最常用的API。
類似于LocalDate和LocalTime從系統(tǒng)時(shí)鐘獲取LocalDateTime的實(shí)例:
LocalDateTime.now();
下面的代碼示例解釋了如何使用工廠“of”和“parse”方法創(chuàng)建實(shí)例。結(jié)果將是代表2015年2月20日06:30 AM 的LocalDateTime實(shí)例:
LocalDateTime.of(2015, Month.FEBRUARY, 20, 06, 30); LocalDateTime.parse("2015-02-20T06:30:00");
有一些實(shí)用的API可以支持特定時(shí)間單位的時(shí)間運(yùn)算,例如天,月,年和分鐘。以下代碼示例演示了“加”和“減”方法的用法。這些API的行為與LocalDate和LocalTime中的 API完全相同:
localDateTime.plusDays(1); localDateTime.minusHours(2);
Getter方法可用于提取類似于日期和時(shí)間類的特定單位。鑒于上面的LocalDateTime實(shí)例,下面的代碼示例將返回2月份的月份:
localDateTime.getMonth();
4.使用ZonedDateTime API
當(dāng)我們需要處理時(shí)區(qū)特定的日期和時(shí)間時(shí),Java 8提供了ZonedDateTime 類。ZoneID是用于表示不同區(qū)域的標(biāo)識(shí)符。大約有40個(gè)不同的時(shí)區(qū),使用ZoneID表示它們,如下所示
下面的代碼我們來獲取下“亞洲/上海”時(shí)區(qū):
ZoneId zoneId = ZoneId.of("Aisa/Shanghai");
獲取所有的時(shí)區(qū):
Set<String> allZoneIds = ZoneId.getAvailableZoneIds();
LocalDateTime轉(zhuǎn)化為特定的時(shí)區(qū)中的時(shí)間:
ZonedDateTime zonedDateTime = ZonedDateTime.of(localDateTime, zoneId);
ZonedDateTime提供解析方法來獲取時(shí)區(qū)的特定日期時(shí)間:
ZonedDateTime.parse("2015-05-03T10:15:30+01:00[Aisa/Shanghai]");
使用時(shí)區(qū)的另一種方法是使用OffsetDateTime。OffsetDateTime是具有偏移量的日期時(shí)間的不可變表示形式。此類存儲(chǔ)所有日期和時(shí)間字段,精確到納秒,以及從UTC/格林威治的偏移量。可以使用ZoneOffset創(chuàng)建OffsetDateTime實(shí)例。這里我們創(chuàng)建一個(gè)LocalDateTime來表示2015年2月20日上午6:30:
LocalDateTime localDateTime = LocalDateTime.of(2015, Month.FEBRUARY, 20, 06, 30);
然后我們通過創(chuàng)建ZoneOffset并為LocalDateTime實(shí)例設(shè)置來增加兩個(gè)小時(shí):
ZoneOffset offset = ZoneOffset.of("+02:00"); OffsetDateTime offSetByTwo = OffsetDateTime.of(localDateTime, offset);
我們現(xiàn)在的本地日期時(shí)間為2015-02-20 06:30 +02:00。現(xiàn)在讓我們繼續(xù)討論如何使用Period和Duration類修改日期和時(shí)間值。
5.使用Period和Duration
- Period : 用于計(jì)算兩個(gè)日期(年月日)間隔。
- Duration : 用于計(jì)算兩個(gè)時(shí)間(秒,納秒)間隔。
5.1.使用Period
Period 類被廣泛地用于修改給定的日期的值或者獲取兩個(gè)日期之間的差值:
LocalDate initialDate = LocalDate.parse("2007-05-10"); LocalDate finalDate = initialDate.plus(Period.ofDays(5));
Period 類有各種getter方法,如getYears,getMonths和getDays從獲取值周期對(duì)象。下面的代碼示例返回一個(gè)int值為5,是基于上面示例的逆序操作:
int five = Period.between(finalDate, initialDate).getDays();
該P(yáng)eriod 可以在特定的單元獲得兩個(gè)日期之間的如天或月或數(shù)年,使用ChronoUnit.between:
int five = ChronoUnit.DAYS.between(finalDate , initialDate);
此代碼示例返回五天。讓我們繼續(xù)看看Duration類。
5.2.使用Duration
類似Period ,該Duration類是用來處理時(shí)間。在下面的代碼中,我們創(chuàng)建一個(gè)本地時(shí)間上午6:30,然后加30秒的持續(xù)時(shí)間,以使本地時(shí)間上午6時(shí)三十〇分30秒的:
LocalTime initialTime = LocalTime.of(6, 30, 0); LocalTime finalTime = initialTime.plus(Duration.ofSeconds(30));
兩個(gè)時(shí)刻之間的持續(xù)時(shí)間可以作為持續(xù)時(shí)間或作為特定單位獲得。在第一個(gè)代碼片段中,我們使用Duration類的between()方法來查找finalTime和initialTime之間的時(shí)間差,并以秒為單位返回差異:
int thirty = Duration.between(finalTime, initialTime).getSeconds();
在第二個(gè)例子中,我們使用ChronoUnit類的between()方法來執(zhí)行相同的操作:
int thirty = ChronoUnit.SECONDS.between(finalTime, initialTime);
現(xiàn)在我們來看看如何將舊的Date 和Calendar 轉(zhuǎn)換為新的Date和Time。
6.與日期和日歷的兼容性
Java 8添加了toInstant()方法,該方法有助于將舊API中的Date和Calendar實(shí)例轉(zhuǎn)換為新的Date Time API,如下面的代碼片段所示:
LocalDateTime.ofInstant(date.toInstant(), ZoneId.systemDefault()); LocalDateTime.ofInstant(calendar.toInstant(), ZoneId.systemDefault());
所述LocalDateTime可以從如下“ofEpochSecond"方法來構(gòu)造。以下代碼的結(jié)果將是代表2016-06-13T11:34:50 的LocalDateTime:
LocalDateTime.ofEpochSecond(1465817690, 0, ZoneOffset.UTC);
現(xiàn)在讓我們繼續(xù)進(jìn)行日期和時(shí)間格式化。
7. 日期和時(shí)間格式化
Java 8提供了用于輕松格式化日期和時(shí)間的 API :
LocalDateTime localDateTime = LocalDateTime.of(2015, Month.JANUARY, 25, 6, 30);
以下代碼傳遞ISO日期格式以格式化本地日期。結(jié)果將是2015-01-25:
String localDateString = localDateTime.format(DateTimeFormatter.ISO_DATE);
該DateTimeFormatter提供多種標(biāo)準(zhǔn)格式選項(xiàng)。也可以提供自定義模式來格式化方法,如下所示,它將返回LocalDate為2015/01/25:
localDateTime.format(DateTimeFormatter.ofPattern("yyyy/MM/dd"));
我們可以將格式樣式傳遞為SHORT,LONG或MEDIUM作為格式化選項(xiàng)的一部分。
下面的代碼示例輸出2015年1月25日06:30:00 me的輸:
localDateTime .format(DateTimeFormatter.ofLocalizedDateTime(FormatStyle.MEDIUM) .withLocale(Locale.UK);
最后讓我們看看Java 8 Core Date / Time API 可用的替代方案。
8.替代方案
8.1.使用Threeten 類庫
對(duì)于從Java 7或Java 6這些老項(xiàng)目來說可以使用Threeten ,然后可以像在上面java 8一樣使用相同的功能,一旦你遷移到j(luò)ava 8 只需要修改你的包路徑代碼而無需變更:
<dependency> <groupId>org.threeten</groupId> <artifactId>threetenbp</artifactId> <version>LATEST</version> </dependency>
8.2.Joda-Time類庫
Java 8 日期和時(shí)間庫的另一種替代方案是Joda-Time庫。事實(shí)上,Java 8 Date Time API由Joda-Time庫(Stephen Colebourne)和Oracle共同領(lǐng)導(dǎo)。該庫提供了Java 8 Date Time項(xiàng)目中支持的幾乎所有功能。通過在項(xiàng)目中引用以下pom依賴項(xiàng)就可以立即使用:
<dependency> <groupId>joda-time</groupId> <artifactId>joda-time</artifactId> <version>LATEST</version> </dependency>
到此這篇關(guān)于Java 8 Time Api 使用方法技巧的文章就介紹到這了,更多相關(guān)Java 8 Time Api 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java編程實(shí)現(xiàn)時(shí)間和時(shí)間戳相互轉(zhuǎn)換實(shí)例
這篇文章主要介紹了什么是時(shí)間戳,以及Java編程實(shí)現(xiàn)時(shí)間和時(shí)間戳相互轉(zhuǎn)換實(shí)例,具有一定的參考價(jià)值,需要的朋友可以了解下。2017-09-09spring集成mybatis實(shí)現(xiàn)mysql數(shù)據(jù)庫讀寫分離
本文通過實(shí)例代碼給大家介紹了spring集成mybatis實(shí)現(xiàn)mysql數(shù)據(jù)庫讀寫分離,需要的朋友可以參考下2017-08-08Java里得到00:00:00格式的時(shí)分秒的Timestamp
Java里如何得到00:00:00格式的時(shí)分秒的Timestamp ,下面是具體的實(shí)現(xiàn)代碼,需要的朋友可以參考下。2009-09-09Java獲取resources下文件路徑的幾種方法及遇到的問題
這篇文章主要給大家介紹了關(guān)于Java獲取resources下文件路徑的幾種方法及遇到的問題,在Java開發(fā)中經(jīng)常需要讀取項(xiàng)目中resources目錄下的文件或獲取資源路徑,需要的朋友可以參考下2023-12-12如何開啟控制臺(tái)輸出mybatis執(zhí)行的sql日志問題
這篇文章主要介紹了如何開啟控制臺(tái)輸出mybatis執(zhí)行的sql日志問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09Java8學(xué)習(xí)教程之lambda表達(dá)式語法介紹
眾所周知lambda表達(dá)式是JAVA8中提供的一種新的特性,它支持Java也能進(jìn)行簡單的“函數(shù)式編程”。 下面這篇文章主要給大家介紹了關(guān)于Java8學(xué)習(xí)教程之lambda表達(dá)式語法的相關(guān)資料,需要的朋友可以參考下。2017-09-09SpringBoot中mapper.xml文件存放的兩種實(shí)現(xiàn)位置
這篇文章主要介紹了SpringBoot中mapper.xml文件存放的兩種實(shí)現(xiàn)位置,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01