死磕18個(gè)Java8日期處理,工作必用!
Java 8 推出了全新的日期時(shí)間API,在教程中我們將通過(guò)一些簡(jiǎn)單的實(shí)例來(lái)學(xué)習(xí)如何使用新API。
Java處理日期、日歷和時(shí)間的方式一直為社區(qū)所詬病,將 java.util.Date設(shè)定為可變類(lèi)型,以及SimpleDateFormat的非線(xiàn)程安全使其應(yīng)用非常受限。
新API基于ISO標(biāo)準(zhǔn)日歷系統(tǒng),java.time包下的所有類(lèi)都是不可變類(lèi)型而且線(xiàn)程安全。
示例1:Java 8中獲取今天的日期
Java 8 中的 LocalDate 用于表示當(dāng)天日期。和java.util.Date不同,它只有日期,不包含時(shí)間。當(dāng)你僅需要表示日期時(shí)就用這個(gè)類(lèi)。
- package com.shxt.demo02;
- import java.time.LocalDate;
- public class Demo01 {
- public static void main(String[] args) {
- LocalDate today = LocalDate.now();
- System.out.println("今天的日期:"+today);
- }
- }
示例2:Java 8中獲取年、月、日信息
- package com.shxt.demo02;
- import java.time.LocalDate;
- public class Demo02 {
- public static void main(String[] args) {
- LocalDate today = LocalDate.now();
- int year = today.getYear();
- int month = today.getMonthValue();
- int day = today.getDayOfMonth();
- System.out.println("year:"+year);
- System.out.println("month:"+month);
- System.out.println("day:"+day);
- }
- }
示例3:Java 8中處理特定日期
我們通過(guò)靜態(tài)工廠(chǎng)方法now()非常容易地創(chuàng)建了當(dāng)天日期,你還可以調(diào)用另一個(gè)有用的工廠(chǎng)方法LocalDate.of()創(chuàng)建任意日期, 該方法需要傳入年、月、日做參數(shù),返回對(duì)應(yīng)的LocalDate實(shí)例。這個(gè)方法的好處是沒(méi)再犯老API的設(shè)計(jì)錯(cuò)誤,比如年度起始于1900,月份是從0開(kāi) 始等等。
- package com.shxt.demo02;
- import java.time.LocalDate;
- public class Demo03 {
- public static void main(String[] args) {
- LocalDate date = LocalDate.of(2018,2,6);
- System.out.println("自定義日期:"+date);
- }
- }
示例4:Java 8中判斷兩個(gè)日期是否相等
- package com.shxt.demo02;
- import java.time.LocalDate;
- public class Demo04 {
- public static void main(String[] args) {
- LocalDate date1 = LocalDate.now();
- LocalDate date2 = LocalDate.of(2018,2,5);
- if(date1.equals(date2)){
- System.out.println("時(shí)間相等");
- }else{
- System.out.println("時(shí)間不等");
- }
- }
- }
示例5:Java 8中檢查像生日這種周期性事件
- package com.shxt.demo02;
- import java.time.LocalDate;
- import java.time.MonthDay;
- public class Demo05 {
- public static void main(String[] args) {
- LocalDate date1 = LocalDate.now();
- LocalDate date2 = LocalDate.of(2018,2,6);
- MonthDay birthday = MonthDay.of(date2.getMonth(),date2.getDayOfMonth());
- MonthDay currentMonthDay = MonthDay.from(date1);
- if(currentMonthDay.equals(birthday)){
- System.out.println("是你的生日");
- }else{
- System.out.println("你的生日還沒(méi)有到");
- }
- }
- }
只要當(dāng)天的日期和生日匹配,無(wú)論是哪一年都會(huì)打印出祝賀信息。你可以把程序整合進(jìn)系統(tǒng)時(shí)鐘,看看生日時(shí)是否會(huì)受到提醒,或者寫(xiě)一個(gè)單元測(cè)試來(lái)檢測(cè)代碼是否運(yùn)行正確。
示例6:Java 8中獲取當(dāng)前時(shí)間
- package com.shxt.demo02;
- import java.time.LocalTime;
- public class Demo06 {
- public static void main(String[] args) {
- LocalTime time = LocalTime.now();
- System.out.println("獲取當(dāng)前的時(shí)間,不含有日期:"+time);
- }
- }
可以看到當(dāng)前時(shí)間就只包含時(shí)間信息,沒(méi)有日期
示例7:Java 8中獲取當(dāng)前時(shí)間
通過(guò)增加小時(shí)、分、秒來(lái)計(jì)算將來(lái)的時(shí)間很常見(jiàn)。Java 8除了不變類(lèi)型和線(xiàn)程安全的好處之外,還提供了更好的plusHours()方法替換add(),并且是兼容的。注意,這些方法返回一個(gè)全新的LocalTime實(shí)例,由于其不可變性,返回后一定要用變量賦值。
- package com.shxt.demo02;
- import java.time.LocalTime;
- public class Demo07 {
- public static void main(String[] args) {
- LocalTime time = LocalTime.now();
- LocalTime newTime = time.plusHours(3);
- System.out.println("三個(gè)小時(shí)后的時(shí)間為:"+newTime);
- }
- }
示例8:Java 8如何計(jì)算一周后的日期
和上個(gè)例子計(jì)算3小時(shí)以后的時(shí)間類(lèi)似,這個(gè)例子會(huì)計(jì)算一周后的日期。LocalDate日期不包含時(shí)間信息,它的plus()方法用來(lái)增加天、周、月,ChronoUnit類(lèi)聲明了這些時(shí)間單位。由于LocalDate也是不變類(lèi)型,返回后一定要用變量賦值。
- package com.shxt.demo02;
- import java.time.LocalDate;
- import java.time.temporal.ChronoUnit;
- public class Demo08 {
- public static void main(String[] args) {
- LocalDate today = LocalDate.now();
- System.out.println("今天的日期為:"+today);
- LocalDate nextWeek = today.plus(1, ChronoUnit.WEEKS);
- System.out.println("一周后的日期為:"+nextWeek);
- }
- }
可以看到新日期離當(dāng)天日期是7天,也就是一周。你可以用同樣的方法增加1個(gè)月、1年、1小時(shí)、1分鐘甚至一個(gè)世紀(jì),更多選項(xiàng)可以查看Java 8 API中的ChronoUnit類(lèi)
示例9:Java 8計(jì)算一年前或一年后的日期
利用minus()方法計(jì)算一年前的日期
- package com.shxt.demo02;
- import java.time.LocalDate;
- import java.time.temporal.ChronoUnit;
- public class Demo09 {
- public static void main(String[] args) {
- LocalDate today = LocalDate.now();
- LocalDate previousYear = today.minus(1, ChronoUnit.YEARS);
- System.out.println("一年前的日期 : " + previousYear);
- LocalDate nextYear = today.plus(1, ChronoUnit.YEARS);
- System.out.println("一年后的日期:"+nextYear);
- }
- }
示例10:Java 8的Clock時(shí)鐘類(lèi)
Java 8增加了一個(gè)Clock時(shí)鐘類(lèi)用于獲取當(dāng)時(shí)的時(shí)間戳,或當(dāng)前時(shí)區(qū)下的日期時(shí)間信息。以前用到System.currentTimeInMillis()和TimeZone.getDefault()的地方都可用Clock替換。
- package com.shxt.demo02;
- import java.time.Clock;
- public class Demo10 {
- public static void main(String[] args) {
- // Returns the current time based on your system clock and set to UTC.
- Clock clock = Clock.systemUTC();
- System.out.println("Clock : " + clock.millis());
- // Returns time based on system clock zone
- Clock defaultClock = Clock.systemDefaultZone();
- System.out.println("Clock : " + defaultClock.millis());
- }
- }
示例11:如何用Java判斷日期是早于還是晚于另一個(gè)日期
另一個(gè)工作中常見(jiàn)的操作就是如何判斷給定的一個(gè)日期是大于某天還是小于某天?在Java 8中,LocalDate類(lèi)有兩類(lèi)方法isBefore()和isAfter()用于比較日期。調(diào)用isBefore()方法時(shí),如果給定日期小于當(dāng)前日期則返回true。
- package com.shxt.demo02;
- import java.time.LocalDate;
- import java.time.temporal.ChronoUnit;
- public class Demo11 {
- public static void main(String[] args) {
- LocalDate today = LocalDate.now();
- LocalDate tomorrow = LocalDate.of(2018,2,6);
- if(tomorrow.isAfter(today)){
- System.out.println("之后的日期:"+tomorrow);
- }
- LocalDate yesterday = today.minus(1, ChronoUnit.DAYS);
- if(yesterday.isBefore(today)){
- System.out.println("之前的日期:"+yesterday);
- }
- }
- }
示例12:Java 8中處理時(shí)區(qū)
Java 8不僅分離了日期和時(shí)間,也把時(shí)區(qū)分離出來(lái)了。現(xiàn)在有一系列單獨(dú)的類(lèi)如ZoneId來(lái)處理特定時(shí)區(qū),ZoneDateTime類(lèi)來(lái)表示某時(shí)區(qū)下的時(shí)間。這在Java 8以前都是 GregorianCalendar類(lèi)來(lái)做的。下面這個(gè)例子展示了如何把本時(shí)區(qū)的時(shí)間轉(zhuǎn)換成另一個(gè)時(shí)區(qū)的時(shí)間。
- package com.shxt.demo02;
- import java.time.LocalDateTime;
- import java.time.ZoneId;
- import java.time.ZonedDateTime;
- public class Demo12 {
- public static void main(String[] args) {
- // Date and time with timezone in Java 8
- ZoneId america = ZoneId.of("America/New_York");
- LocalDateTime localtDateAndTime = LocalDateTime.now();
- ZonedDateTime dateAndTimeInNewYork = ZonedDateTime.of(localtDateAndTime, america );
- System.out.println("Current date and time in a particular timezone : " + dateAndTimeInNewYork);
- }
- }
示例13:如何表示信用卡到期這類(lèi)固定日期,答案就在YearMonth
與 MonthDay檢查重復(fù)事件的例子相似,YearMonth是另一個(gè)組合類(lèi),用于表示信用卡到期日、FD到期日、期貨期權(quán)到期日等。還可以用這個(gè)類(lèi)得到 當(dāng)月共有多少天,YearMonth實(shí)例的lengthOfMonth()方法可以返回當(dāng)月的天數(shù),在判斷2月有28天還是29天時(shí)非常有用。
- package com.shxt.demo02;
- import java.time.*;
- public class Demo13 {
- public static void main(String[] args) {
- YearMonth currentYearMonth = YearMonth.now();
- System.out.printf("Days in month year %s: %d%n", currentYearMonth, currentYearMonth.lengthOfMonth());
- YearMonth creditCardExpiry = YearMonth.of(2019, Month.FEBRUARY);
- System.out.printf("Your credit card expires on %s %n", creditCardExpiry);
- }
- }
示例14:如何在Java 8中檢查閏年
- package com.shxt.demo02;
- import java.time.LocalDate;
- public class Demo14 {
- public static void main(String[] args) {
- LocalDate today = LocalDate.now();
- if(today.isLeapYear()){
- System.out.println("This year is Leap year");
- }else {
- System.out.println("2018 is not a Leap year");
- }
- }
- }
示例15:計(jì)算兩個(gè)日期之間的天數(shù)和月數(shù)
有一個(gè)常見(jiàn)日期操作是計(jì)算兩個(gè)日期之間的天數(shù)、周數(shù)或月數(shù)。在Java 8中可以用java.time.Period類(lèi)來(lái)做計(jì)算。
下面這個(gè)例子中,我們計(jì)算了當(dāng)天和將來(lái)某一天之間的月數(shù)。
- package com.shxt.demo02;
- import java.time.LocalDate;
- import java.time.Period;
- public class Demo15 {
- public static void main(String[] args) {
- LocalDate today = LocalDate.now();
- LocalDate java8Release = LocalDate.of(2018, 12, 14);
- Period periodToNextJavaRelease = Period.between(today, java8Release);
- System.out.println("Months left between today and Java 8 release : "
- + periodToNextJavaRelease.getMonths() );
- }
- }
示例16:在Java 8中獲取當(dāng)前的時(shí)間戳
Instant類(lèi)有一個(gè)靜態(tài)工廠(chǎng)方法now()會(huì)返回當(dāng)前的時(shí)間戳,如下所示:
- package com.shxt.demo02;
- import java.time.Instant;
- public class Demo16 {
- public static void main(String[] args) {
- Instant timestamp = Instant.now();
- System.out.println("What is value of this instant " + timestamp.toEpochMilli());
- }
- }
時(shí)間戳信息里同時(shí)包含了日期和時(shí)間,這和java.util.Date很像。實(shí)際上Instant類(lèi)確實(shí)等同于 Java 8之前的Date類(lèi),你可以使用Date類(lèi)和Instant類(lèi)各自的轉(zhuǎn)換方法互相轉(zhuǎn)換,例如:Date.from(Instant) 將Instant轉(zhuǎn)換成java.util.Date,Date.toInstant()則是將Date類(lèi)轉(zhuǎn)換成Instant類(lèi)。
示例17:Java 8中如何使用預(yù)定義的格式化工具去解析或格式化日期
- package com.shxt.demo02;
- import java.time.LocalDate;
- import java.time.format.DateTimeFormatter;
- public class Demo17 {
- public static void main(String[] args) {
- String dayAfterTommorrow = "20180205";
- LocalDate formatted = LocalDate.parse(dayAfterTommorrow,
- DateTimeFormatter.BASIC_ISO_DATE);
- System.out.println(dayAfterTommorrow+" 格式化后的日期為: "+formatted);
- }
- }
示例18:字符串互轉(zhuǎn)日期類(lèi)型
- package com.shxt.demo02;
- import java.time.LocalDate;
- import java.time.LocalDateTime;
- import java.time.format.DateTimeFormatter;
- public class Demo18 {
- public static void main(String[] args) {
- LocalDateTime date = LocalDateTime.now();
- DateTimeFormatter format1 = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss");
- //日期轉(zhuǎn)字符串
- String str = date.format(format1);
- System.out.println("日期轉(zhuǎn)換為字符串:"+str);
- DateTimeFormatter format2 = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss");
- //字符串轉(zhuǎn)日期
- LocalDate date2 = LocalDate.parse(str,format2);
- System.out.println("日期類(lèi)型:"+date2);
- }
- }