Java 8 Time API

来源:https://www.cnblogs.com/ssymon/archive/2022/10/26/16829643.html
-Advertisement-
Play Games

Java 8 系列文章 持續更新中 日期時間API 也是Java 8重要的更新之一,Java從一開始就缺少一致的日期和時間方法,Java 8 Date Time API是Java核心API的一個非常好的補充。 為什麼需要新的日期時間API Java中現有的與日期和時間相關的類存在一些問題: 日期時間 ...


Java 8 系列文章 持續更新中

日期時間API 也是Java 8重要的更新之一,Java從一開始就缺少一致的日期和時間方法,Java 8 Date Time API是Java核心API的一個非常好的補充。

為什麼需要新的日期時間API

Java中現有的與日期和時間相關的類存在一些問題:

  • 日期時間類的定義不一致,在java.utiljava.sql包中都有Date類。同樣,格式化和解析類是在java.text包中定義的。
  • java.util.Date同時包含日期和時間值,而java.sql.Date只包含日期值,把它放在java.sql包中是沒有意義的。而且這兩個類的名稱相同,這本身就是一個非常糟糕的設計。
  • 沒有為時間、時間戳、格式化和解析明確定義的類。我們有java.text.DateFormat抽象類用於解析和格式化,通常使用SimpleDateFormat類解析和格式化。
  • 所有Date類都是可變的,所以它們不是線程安全的,這也是JavaDateCalendar類最大的問題之一。
  • Date類不提供國際化,不支持時區。雖然引入了java.util.Calendarjava.util.TimeZone,但是它們也存在上面的問題。

DateCalendar類中定義的方法還有一些其他的問題,但是上面的問題清楚地表明,Java中需要一個健壯的日期時間API。這就是為什麼 Joda Time 可以成為Java 日期時間高質量的替代品。

Java 8日期時間設計原則

Java 8日期時間API是基於 JSR-310 規範實現的。目的是為瞭解決遺留日期時間實現中的所有缺陷。新的日期時間API的一些設計原則如下:

  • 不變性:新的日期時間API中的所有類都是不可變的,適用於多線程環境。

  • 關註點分離:新的日期時間API明確區分了人類可讀的日期、時間和機器時間(Unix時間戳),它為DateTimeDateTimeTimestampTimezone 等定義單獨的類。

  • 清晰性:所有的類中都清晰地定義了方法,並執行相同的操作。例如,要獲取當前時間實例可以用now()方法,在所有這些類中都定義了format()parse()方法,而不是為它們單獨定義一個類。

    所有類都使用工廠模式策略模式來更好地操作。一旦您使用了其中一個類中的方法,使用其他類並不困難。

  • 實用的操作:所有新的日期時間API類都常見的方法,比如加、減、格式化、解析、在日期/時間中獲取單獨的部分等等。

  • 可擴展:新的日期時間API可以在ISO-8601日曆系統上工作,但是我們也可以在其他非ISO日曆上使用它。

Java 8日期時間API的包

Java8日期時間API由以下包組成。

  • java.time:這是Java 8日期時間API的基本包。所有主要的基類都是這個包的一部分,例如LocalDateLocalTimeLocalDateTimeInstantPeriodDuration等。所有這些類都是不可變的和線程安全的。大多數情況下,這些類足以處理常見的需求。
  • java.time.chrono:這個包為非ISO日曆系統定義了通用API。我們可以擴展AbstractChronology類來創建我們自己的日曆系統。
  • java.time.format:這個包包含用於格式化和解析日期時間對象的類。大多數時候我們不會直接使用它們,因為java.time包中的日期時間類已經提供了格式化和解析方法。
  • java.time.temporal:這個包包含temporal對象,我們可以使用它來找出與日期/時間對象相關的特定日期或時間。例如,我們可以使用它們來查找一個月的第一天或最後一天。您可以很容易地識別這些方法,因為它們的格式總是withXXX
  • java.time.zone:這個包包含用於支持不同時區及其規則的類。

Java 8日期時間API類的示例

下麵通過一些日期時間API類的示例,來更好的瞭解Java 8日期時間API

1.LocalDate

LocalDate是一個不可變的日期類,它以yyyy-MM-dd的預設格式表示日期。可以使用now()方法來獲取當前日期,還可以提供年、月和日期的輸入參數來創建LocalDate實例。

這個類為now()提供了一個重載方法,在這裡可以傳遞ZoneId來獲取特定時區中的日期。這個類提供了與java.sql.Date相同的功能。

// 當前日期
LocalDate today = LocalDate.now();
System.out.println("當前日期=" + today);

// 通過提供年月日參數創建日期
LocalDate nowYear_1024 = LocalDate.of(2022, Month.OCTOBER, 24);
System.out.println("參數日期=" + nowYear_1024);

// 通過時區獲取當前日期
LocalDate todayShanghai = LocalDate.now(ZoneId.of("Asia/Shanghai"));
System.out.println("當前日期(CTT)=" + todayShanghai);

// 從紀元日(1970-01-01)開始的第多少天
LocalDate dateFromBase = LocalDate.ofEpochDay(365);
System.out.println("1970-01-01的第365天= " + dateFromBase);

// 某年的第多少天
LocalDate hundredDay2022 = LocalDate.ofYearDay(2022, 100);
System.out.println("2022年的第100天=" + hundredDay2022);

運行之後結果如下:

當前日期=2022-10-26
特殊日期=2022-10-24
當前日期(CTT)=2022-10-26
1970-01-01的第365天= 1971-01-01
2022年的第100天=2022-04-10

2.LocalTime

LocalTime是一個不可變的時間類,它以HH:mm:ss.SSS的預設格式表示時間。與LocalDate一樣,這個類提供了時區支持,並可以通過傳遞小時、分鐘和秒作為輸入參數來創建實例。

// 當前時間
LocalTime time = LocalTime.now();
System.out.println("當前時間=" + time);

// 通過提供時分秒參數創建日期
LocalTime specificTime = LocalTime.of(12, 20, 25, 40);
System.out.println("參數時間=" + specificTime);

// 通過時區獲取當前時間
LocalTime timeShanghai = LocalTime.now(ZoneId.of("Asia/Shanghai"));
System.out.println("當前時間(CTT)=" + timeShanghai);

// 從紀元日開始的第多少秒
LocalTime specificSecondTime = LocalTime.ofSecondOfDay(100);
System.out.println("從紀元日開始的第100秒=" + specificSecondTime);

運行之後結果如下:

當前時間=15:39:18.948
參數時間=12:20:25.000000040
當前時間(CTT)=15:39:18.949
從0開始的第100秒=00:01:40

3.LocalDateTime

LocalDateTime是一個不可變的日期時間類,它以yyyy-MM-ddTHH:mm:ss.SSS的預設格式表示時間日期。它提供了一個工廠方法,該方法使用LocalDateLocalTime作為參數創建LocalDateTime實例。

// 當前日期時間
LocalDateTime now = LocalDateTime.now();
System.out.println("當前日期時間=" + now);

// 通過提供LocalDate和LocalTime參數創建日期時間
now = LocalDateTime.of(LocalDate.now(), LocalTime.now());
System.out.println("當前日期時間=" + now);

// 通過提供年月日時分秒參數創建日期時間
LocalDateTime specificTime = LocalDateTime.of(2022, Month.OCTOBER, 24, 10, 24, 24);
System.out.println("參數日期時間=" + specificTime);

// 通過時區獲取當前日期時間
LocalDateTime todayKolkata = LocalDateTime.now(ZoneId.of("Asia/Shanghai"));
System.out.println("當前日期時間(CTT)=" + todayKolkata);

// 從紀元日開始的第多少秒
LocalDateTime dateFromBase = LocalDateTime.ofEpochSecond(100, 0, ZoneOffset.UTC);
System.out.println("從紀元日開始的第100秒= " + dateFromBase);

運行之後結果如下:

當前日期時間=2022-10-26T15:51:59.070
當前日期時間=2022-10-26T15:51:59.071
參數日期時間=2022-10-24T10:24:24
當前日期時間(CTT)=2022-10-26T15:51:59.071
從紀元日開始的第100秒=1970-01-01T00:01:40

註意:以上例子通過輸入參數創建實例時,如果輸入了無效的參數name將會拋出java.time.DateTimeException

4.Instant

instant類用於處理機器可讀的時間格式。instant類將日期時間存儲在unix時間戳中。

// 當期時間戳
Instant timestamp = Instant.now();
System.out.println("當期時間戳= "+timestamp);

// 從紀元日開始的第多少毫秒
Instant specificTime = Instant.ofEpochMilli(timestamp.toEpochMilli());
System.out.println("從紀元日開始="+specificTime);

運行之後結果如下:

當期時間戳=2022-10-26T08:08:40.429Z
從紀元日開始=2022-10-26T08:08:40.429Z

Java8日期時間API類的實用方法

大多數日期時間類都會提供各種實用方法,例如加/減天數、周數、月數等。還有一些其他實用方法可以使用時間調整器TemporalAdjuster調整日期,並計算兩個日期之間的時間段。

LocalDate today = LocalDate.now();

//獲取年份,判斷年份是否是閏年
System.out.println("Year "+today.getYear()+" is Leap Year? "+today.isLeapYear());

//比較兩個時間
System.out.println("Today is before 01/01/2023? "+today.isBefore(LocalDate.of(2023,1,1)));

//通過LocalDate創建LocalDateTime
System.out.println("Current Time="+today.atTime(LocalTime.now()));

//加減操作
System.out.println("10 days after today will be "+today.plusDays(10));
System.out.println("3 weeks after today will be "+today.plusWeeks(3));
System.out.println("20 months after today will be "+today.plusMonths(20));

System.out.println("10 days before today will be "+today.minusDays(10));
System.out.println("3 weeks before today will be "+today.minusWeeks(3));
System.out.println("20 months before today will be "+today.minusMonths(20));

//時間調整器調整時間
System.out.println("First date of this month= "+today.with(TemporalAdjusters.firstDayOfMonth()));
LocalDate lastDayOfYear = today.with(TemporalAdjusters.lastDayOfYear());
System.out.println("Last date of this year= "+lastDayOfYear);

Period period = today.until(lastDayOfYear);
System.out.println("Period Format= "+period);
System.out.println("Months remaining in the year= "+period.getMonths());

運行之後結果如下:

Year 2022 is Leap Year? false
Today is before 01/01/2023? true
Current Time=2022-10-26T16:25:04.740
10 days after today will be 2022-11-05
3 weeks after today will be 2022-11-16
20 months after today will be 2024-06-26
10 days before today will be 2022-10-16
3 weeks before today will be 2022-10-05
20 months before today will be 2021-02-26
First date of this month= 2022-10-01
Last date of this year= 2022-12-31
Period Format= P2M5D
Months remaining in the year= 2

Java8日期時間的解析和格式化

經常用到的操作有:將日期時間格式化為不同格式String,解析String以獲得日期時間對象。

// 格式化
LocalDate date = LocalDate.now();
// 預設格式
System.out.println("Default format of LocalDate=" + date);
// 自定義格式
System.out.println(date.format(DateTimeFormatter.ofPattern("yyyy年MM月dd日")));
System.out.println(date.format(DateTimeFormatter.BASIC_ISO_DATE));


LocalDateTime dateTime = LocalDateTime.now();
// 預設格式
System.out.println("Default format of LocalDateTime=" + dateTime);
// 自定義格式
System.out.println(dateTime.format(DateTimeFormatter.ofPattern("yyyy年MM月dd日HH時mm分ss秒")));
System.out.println(dateTime.format(DateTimeFormatter.BASIC_ISO_DATE));

Instant timestamp = Instant.now();
// 預設格式
System.out.println("Default format of Instant=" + timestamp);

// 解析
LocalDateTime dt = LocalDateTime.parse("2022年10月24日10時24分24秒",
        DateTimeFormatter.ofPattern("yyyy年MM月dd日HH時mm分ss秒"));
System.out.println("Default format after parsing = " + dt);

運行之後結果如下:

Default format of LocalDate=2022-10-26
2022年10月26日
20221026
Default format of LocalDateTime=2022-10-26T16:37:51.300
2022年10月26日16時37分51秒
20221026
Default format of Instant=2022-10-26T08:37:51.301Z
Default format after parsing = 2022-10-24T10:24:24

對遺留日期時間的支持

遺留日期/時間類幾乎在所有應用程式中都使用,因此必須有向下相容。這就是為什麼我們可以通過一些實用方法將遺留類轉換為新類,反之亦然。

//Date轉Instant
Instant timestamp = new Date().toInstant();
//Instant轉LocalDateTime
LocalDateTime date = LocalDateTime.ofInstant(timestamp,
        ZoneId.of(ZoneId.SHORT_IDS.get("CTT")));
System.out.println("Date = " + date);

//Calendar轉Instant
Instant time = Calendar.getInstance().toInstant();
System.out.println(time);
//TimeZone轉ZoneId
ZoneId defaultZone = TimeZone.getDefault().toZoneId();
System.out.println(defaultZone);

//ZonedDateTime from specific Calendar
ZonedDateTime gregorianCalendarDateTime = new GregorianCalendar().toZonedDateTime();
System.out.println(gregorianCalendarDateTime);

//Date API to Legacy classes
Date dt = Date.from(Instant.now());
System.out.println(dt);

TimeZone tz = TimeZone.getTimeZone(defaultZone);
System.out.println(tz);

GregorianCalendar gc = GregorianCalendar.from(gregorianCalendarDateTime);
System.out.println(gc);

運行之後結果如下:

Date = 2022-10-26T16:47:38.329
2022-10-26T08:47:38.429Z
Asia/Shanghai
2022-10-26T16:47:38.455+08:00[Asia/Shanghai]
Wed Oct 26 16:47:38 CST 2022
sun.util.calendar.ZoneInfo[id="Asia/Shanghai",offset=28800000,dstSavings=0,useDaylight=false,transitions=31,lastRule=null]
java.util.GregorianCalendar[time=1666774058455,areFieldsSet=true,areAllFieldsSet=true,lenient=true,zone=sun.util.calendar.ZoneInfo[id="Asia/Shanghai",offset=28800000,dstSavings=0,useDaylight=false,transitions=31,lastRule=null],firstDayOfWeek=2,minimalDaysInFirstWeek=4,ERA=1,YEAR=2022,MONTH=9,WEEK_OF_YEAR=43,WEEK_OF_MONTH=4,DAY_OF_MONTH=26,DAY_OF_YEAR=299,DAY_OF_WEEK=4,DAY_OF_WEEK_IN_MONTH=4,AM_PM=1,HOUR=4,HOUR_OF_DAY=16,MINUTE=47,SECOND=38,MILLISECOND=455,ZONE_OFFSET=28800000,DST_OFFSET=0]

可以看到,遺留的TimeZoneGregorianCalendartoString()方法過於冗長,對用戶不友好。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 01、<form>表單 <form> 表單是比較重要的HTML元素,塊元素,主要作用是向服務端提交數據。結合表單元素input使用,通過內部的button按鈕提交(type="submit")表單數據。 | 元素/屬性 | 描述 | 值/備註 | | | | | | <form> | 表單元素 | ...
  • 今天看到一篇文章中提到了一個好用的工具release-it。剛好可以用在我正在開發的vue3組件庫。紙上得來終覺淺,絕知此事要躬行,說乾就乾,下麵就介紹如何將release-it應用到實際項目中,讓組件庫可以自動化發佈、管理版本號、生成 changelog、tag等 項目調整 在使用這個工具之前先對 ...
  • 什麼是分散式鎖?對於這個問題,相信很多同學是既熟悉又陌生。隨著分散式系統的快速發展與廣泛應用,針對共用資源的互斥訪問也成為了很多業務必須要面對的需求,這個場景下人們通常會引入分散式鎖來解決問題。我們通常會使用怎麼樣的分佈鎖服務呢?有開源的 MySQL,Redis,ZooKeeper,Etcd 等三方... ...
  • 模板方法模式是一種行為設計模式,它在超類中定義了一個演算法的框架,允許子類在不修改結構的情況下重寫演算法的特定步驟。 ...
  • 您好,我是湘王,這是我的博客園,歡迎您來,歡迎您再來~ 為了提高CPU的利用率,工程師們創造了多線程。但是線程們說:要有光!(為了減少線程創建(T1啟動)和銷毀(T3切換)的時間),於是工程師們又接著創造了線程池ThreadPool。就這樣就可以了嗎?——不,工程師們並不滿足於此,他們不把自己創造出 ...
  • 之所以提這個坑,是因為,今天下午,通過監控系統,發現我們系統生產能力突然下降,頻繁報無法獲取資料庫連接。究其原因,竟然是因為mybatisplus的這個“坑”導致的。 ...
  • 圍棋要實現什麼功能呢? 首先是黑棋和白棋,下麵的代碼,黑棋占領的位置被賦值為1,白棋是2; 其次有幾個圍棋中的規則(不知道是不是這樣的,老師是這麼告訴我的,可以指正):1.塊:上下左右可以連起來的叫塊,一顆棋上下左右四個方向若有棋子,就可以稱作是一塊,一個棋子上下左右都沒有,也可以稱作一塊。2.氣: ...
  • 前言 嗨嘍~大家好呀,這裡是魔王吶 ! 網上有很多的創意二維碼,看了,別的不說 羡慕肯定是有的,羡慕有了這不得自己整點活~ 今天我們就來試試只用幾行代碼,生成動態二維碼! 開發環境: Python 3.8 Pycharm 模塊使用: 第三方模塊 需要安裝 在cmd裡面 進行 pip install ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...