18個Java8日期處理的實踐,太有用了!
Java 8 推出了全新的日期時間API,在教程中我們將通過一些簡單的實例來學(xué)習(xí)如何使用新API。
Java處理日期、日歷和時間的方式一直為社區(qū)所詬病,將 java.util.Date設(shè)定為可變類型,以及SimpleDateFormat的非線程安全使其應(yīng)用非常受限。
新API基于ISO標(biāo)準(zhǔn)日歷系統(tǒng),java.time包下的所有類都是不可變類型而且線程安全。
示例1:Java 8中獲取今天的日期
Java 8 中的 LocalDate 用于表示當(dāng)天日期。和java.util.Date不同,它只有日期,不包含時間。當(dāng)你僅需要表示日期時就用這個類。
- 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中處理特定日期
我們通過靜態(tài)工廠方法now()非常容易地創(chuàng)建了當(dāng)天日期,你還可以調(diào)用另一個有用的工廠方法LocalDate.of()創(chuàng)建任意日期, 該方法需要傳入年、月、日做參數(shù),返回對應(yīng)的LocalDate實例。這個方法的好處是沒再犯老API的設(shè)計錯誤,比如年度起始于1900,月份是從0開 始等等。
- 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中判斷兩個日期是否相等
- 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("時間相等");
- }else{
- System.out.println("時間不等");
- }
- }
- }
示例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("你的生日還沒有到");
- }
- }
- }
只要當(dāng)天的日期和生日匹配,無論是哪一年都會打印出祝賀信息。你可以把程序整合進(jìn)系統(tǒng)時鐘,看看生日時是否會受到提醒,或者寫一個單元測試來檢測代碼是否運(yùn)行正確。
示例6:Java 8中獲取當(dāng)前時間
- 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)前的時間,不含有日期:"+time);
- }
- }
可以看到當(dāng)前時間就只包含時間信息,沒有日期
示例7:Java 8中獲取當(dāng)前時間
通過增加小時、分、秒來計算將來的時間很常見。Java 8除了不變類型和線程安全的好處之外,還提供了更好的plusHours()方法替換add(),并且是兼容的。注意,這些方法返回一個全新的LocalTime實例,由于其不可變性,返回后一定要用變量賦值。
- 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("三個小時后的時間為:"+newTime);
- }
- }
示例8:Java 8如何計算一周后的日期
和上個例子計算3小時以后的時間類似,這個例子會計算一周后的日期。LocalDate日期不包含時間信息,它的plus()方法用來增加天、周、月,ChronoUnit類聲明了這些時間單位。由于LocalDate也是不變類型,返回后一定要用變量賦值。
- 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個月、1年、1小時、1分鐘甚至一個世紀(jì),更多選項可以查看Java 8 API中的ChronoUnit類
示例9:Java 8計算一年前或一年后的日期
利用minus()方法計算一年前的日期
- 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時鐘類
Java 8增加了一個Clock時鐘類用于獲取當(dāng)時的時間戳,或當(dāng)前時區(qū)下的日期時間信息。以前用到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判斷日期是早于還是晚于另一個日期
另一個工作中常見的操作就是如何判斷給定的一個日期是大于某天還是小于某天?在Java 8中,LocalDate類有兩類方法isBefore()和isAfter()用于比較日期。調(diào)用isBefore()方法時,如果給定日期小于當(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中處理時區(qū)
Java 8不僅分離了日期和時間,也把時區(qū)分離出來了?,F(xiàn)在有一系列單獨(dú)的類如ZoneId來處理特定時區(qū),ZoneDateTime類來表示某時區(qū)下的時間。這在Java 8以前都是 GregorianCalendar類來做的。下面這個例子展示了如何把本時區(qū)的時間轉(zhuǎn)換成另一個時區(qū)的時間。
- 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:如何表示信用卡到期這類固定日期,答案就在YearMonth
與 MonthDay檢查重復(fù)事件的例子相似,YearMonth是另一個組合類,用于表示信用卡到期日、FD到期日、期貨期權(quán)到期日等。還可以用這個類得到 當(dāng)月共有多少天,YearMonth實例的lengthOfMonth()方法可以返回當(dāng)月的天數(shù),在判斷2月有28天還是29天時非常有用。
- 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:計算兩個日期之間的天數(shù)和月數(shù)
有一個常見日期操作是計算兩個日期之間的天數(shù)、周數(shù)或月數(shù)。在Java 8中可以用java.time.Period類來做計算。
下面這個例子中,我們計算了當(dāng)天和將來某一天之間的月數(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)前的時間戳
Instant類有一個靜態(tài)工廠方法now()會返回當(dāng)前的時間戳,如下所示:
- 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());
- }
- }
時間戳信息里同時包含了日期和時間,這和java.util.Date很像。實際上Instant類確實等同于 Java 8之前的Date類,你可以使用Date類和Instant類各自的轉(zhuǎn)換方法互相轉(zhuǎn)換,例如:Date.from(Instant) 將Instant轉(zhuǎn)換成java.util.Date,Date.toInstant()則是將Date類轉(zhuǎn)換成Instant類。
示例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)日期類型
- 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("日期類型:"+date2);
- }
- }