Java8新特性O(shè)ptional類及新時間日期API示例詳解
Optional類
面試官:Optional類了解過嗎?
這個Optional類主要是解決空指針的問題。
以前對null的處理
@Test public void test01(){ String userName = null; if(userName != null){ System.out.println("字符串的長度:" + userName.length()); }else{ System.out.println("字符串為空"); } }
Optional類介紹
Optional是一個沒有子類的工具類,Optional是一個可以為null的容器對象,它的主要作用就是為了避免Null檢查,防止NullpointerException。
Optional的基本使用
Optional對象的創(chuàng)建方式:
/** * Optional對象的創(chuàng)建方式 */ @Test public void test02(){ // 第一種方式 通過of方法 of方法是不支持null的 Optional<String> op1 = Optional.of("zhangsan"); //Optional<Object> op2 = Optional.of(null); // 第二種方式通過 ofNullable方法 支持null Optional<String> op3 = Optional.ofNullable("lisi"); Optional<Object> op4 = Optional.ofNullable(null); // 第三種方式 通過empty方法直接創(chuàng)建一個空的Optional對象 Optional<Object> op5 = Optional.empty(); }
Optional的常用方法
- get(): 如果Optional有值則返回,否則拋出NoSuchElementException異常。get()通常和isPresent方法一塊使用
- isPresent():判斷是否包含值,包含值返回true,不包含值返回false
- orElse(T t):如果調(diào)用對象包含值,就返回該值,否則返回t
- orElseGet(Supplier s):如果調(diào)用對象包含值,就返回該值,否則返回 Lambda表達(dá)式的返回值
@Test public void test03(){ Optional<String> op1 = Optional.of("zhangsan"); Optional<String> op2 = Optional.empty(); // 獲取Optional中的值 if(op1.isPresent()){ String s1 = op1.get(); System.out.println("用戶名稱:" +s1); } if(op2.isPresent()){ System.out.println(op2.get()); }else{ System.out.println("op2是一個空Optional對象"); } String s3 = op1.orElse("李四"); System.out.println(s3); String s4 = op2.orElse("王五"); System.out.println(s4); String s5 = op2.orElseGet(()->{ return "Hello"; }); System.out.println(s5); } @Test public void test04(){ Optional<String> op1 = Optional.of("zhangsan"); Optional<String> op2 = Optional.empty(); // 如果存在值 就做什么 op1.ifPresent(s-> System.out.println("有值:" +s)); op1.ifPresent(System.out::println); } /** * 自定義一個方法,將Person對象中的 name 轉(zhuǎn)換為大寫 并返回 */ @Test public void test05(){ Person p = new Person("zhangsan",18); Optional<Person> op = Optional.of(p); String name = getNameForOptional(op); System.out.println("name="+name); } /** * 根據(jù)Person對象 將name轉(zhuǎn)換為大寫并返回 * 通過Optional方式實現(xiàn) * @param op * @return */ public String getNameForOptional(Optional<Person> op){ if(op.isPresent()){ String msg = //op.map(p -> p.getName()) op.map(Person::getName) //.map(p -> p.toUpperCase()) .map(String::toUpperCase) .orElse("空值"); return msg; } return null; } /** * 根據(jù)Person對象 將name轉(zhuǎn)換為大寫并返回 * @param person * @return */ public String getName(Person person){ if(person != null){ String name = person.getName(); if(name != null){ return name.toUpperCase(); }else{ return null; } }else{ return null; } }
新時間日期API
面試官:說說Java8新的時間日期API
舊版日期時間的問題
在舊版本中JDK對于日期和時間這塊的時間是非常差的。
- 設(shè)計不合理,在java.util和java.sql的包中都有日期類,java.util.Date同時包含日期和時間的,而java.sql.Date僅僅包含日期,此外用于格式化和解析的類在java.text包下。
- 非線程安全,java.util.Date是非線程安全的,所有的日期類都是可變的,這是java日期類最大的問題之一。
- 時區(qū)處理麻煩,日期類并不提供國際化,沒有時區(qū)支持。
新日期時間API介紹
JDK 8中增加了一套全新的日期時間API,這套API設(shè)計合理,是線程安全的。新的日期及時間API位于 java.time 包 中,下面是一些關(guān)鍵類。
- LocalDate :表示日期,包含年月日,格式為 2019-10-16
- LocalTime :表示時間,包含時分秒,格式為 16:38:54.158549300
- LocalDateTime :表示日期時間,包含年月日,時分秒,格式為 2018-09-06T15:33:56.750
- DateTimeFormatter :日期時間格式化類。
- Instant:時間戳,表示一個特定的時間瞬間。
- Duration:用于計算2個時間(LocalTime,時分秒)的距離
- Period:用于計算2個日期(LocalDate,年月日)的距離
- ZonedDateTime :包含時區(qū)的時間
Java中使用的歷法是ISO 8601日歷系統(tǒng),它是世界民用歷法,也就是我們所說的公歷。平年有365天,閏年是366 天。此外Java 8還提供了4套其他歷法,分別是:
- ThaiBuddhistDate:泰國佛教歷
- MinguoDate:中華國歷
- JapaneseDate:日本歷
- HijrahDate:伊斯蘭歷
日期時間的常見操作
LocalDate,LocalTime以及LocalDateTime的操作。
/** * JDK8 日期時間操作 */ @Test public void test01(){ // 1.創(chuàng)建指定的日期 LocalDate date1 = LocalDate.of(2021, 05, 06); System.out.println("date1 = "+date1); // 2.得到當(dāng)前的日期 LocalDate now = LocalDate.now(); System.out.println("now = "+now); // 3.根據(jù)LocalDate對象獲取對應(yīng)的日期信息 System.out.println("年:" + now.getYear()); System.out.println("月:" + now.getMonth().getValue()); System.out.println("日:" + now.getDayOfMonth()); System.out.println("星期:" + now.getDayOfWeek().getValue()); } /** * 時間操作 */ @Test public void test02(){ // 1.得到指定的時間 LocalTime time = LocalTime.of(5,26,33,23145); System.out.println(time); // 2.獲取當(dāng)前的時間 LocalTime now = LocalTime.now(); System.out.println(now); // 3.獲取時間信息 System.out.println(now.getHour()); System.out.println(now.getMinute()); System.out.println(now.getSecond()); System.out.println(now.getNano()); } /** * 日期時間類型 LocalDateTime */ @Test public void test03(){ // 獲取指定的日期時間 LocalDateTime dateTime = LocalDateTime.of(2020 , 06 , 01 , 12 , 12 , 33 , 213); System.out.println(dateTime); // 獲取當(dāng)前的日期時間 LocalDateTime now = LocalDateTime.now(); System.out.println(now); // 獲取日期時間信息 System.out.println(now.getYear()); System.out.println(now.getMonth().getValue()); System.out.println(now.getDayOfMonth()); System.out.println(now.getDayOfWeek().getValue()); System.out.println(now.getHour()); System.out.println(now.getMinute()); System.out.println(now.getSecond()); System.out.println(now.getNano()); }
日期時間的修改和比較
/** * 日期時間的修改 */ @Test public void test01(){ LocalDateTime now = LocalDateTime.now(); System.out.println("now = "+now); // 修改日期時間 對日期時間的修改,對已存在的LocalDate對象,創(chuàng)建了它模板 // 并不會修改原來的信息 LocalDateTime localDateTime = now.withYear(1998); System.out.println("now :"+now); System.out.println("修改后的:" + localDateTime); System.out.println("月份:" + now.withMonth(10)); System.out.println("天:" + now.withDayOfMonth(6)); System.out.println("小時:" + now.withHour(8)); System.out.println("分鐘:" + now.withMinute(15)); // 在當(dāng)前日期時間的基礎(chǔ)上 加上或者減去指定的時間 System.out.println("兩天后:" + now.plusDays(2)); System.out.println("10年后:"+now.plusYears(10)); System.out.println("6個月后 = " + now.plusMonths(6)); System.out.println("10年前 = " + now.minusYears(10)); System.out.println("半年前 = " + now.minusMonths(6)); System.out.println("一周前 = " + now.minusDays(7)); } /** * 日期時間的比較 */ @Test public void test02(){ LocalDate now = LocalDate.now(); LocalDate date = LocalDate.of(2020, 1, 3); // 在JDK8中要實現(xiàn) 日期的比較 isAfter isBefore isEqual 通過這幾個方法來直接比較 System.out.println(now.isAfter(date)); // true System.out.println(now.isBefore(date)); // false System.out.println(now.isEqual(date)); // false }
注意:在進(jìn)行日期時間修改的時候,原來的LocalDate對象是不會被修改,每次操作都是返回了一個新的LocalDate對象,所以在多線程場景下是數(shù)據(jù)安全的。
格式化和解析操作
在JDK8中我們可以通過java.time.format.DateTimeFormatter
類可以進(jìn)行日期的解析和格式化操作。
@Test public void test01(){ LocalDateTime now = LocalDateTime.now(); // 指定格式 使用系統(tǒng)默認(rèn)的格式 2021-05-27T16:16:38.139 DateTimeFormatter isoLocalDateTime = DateTimeFormatter.ISO_LOCAL_DATE_TIME; // 將日期時間轉(zhuǎn)換為字符串 String format = now.format(isoLocalDateTime); System.out.println("format = " + format); // 通過 ofPattern 方法來指定特定的格式 DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String format1 = now.format(dateTimeFormatter); // 2021-05-27 16:16:38 System.out.println("format1 = " + format1); // 將字符串解析為一個 日期時間類型 LocalDateTime parse = LocalDateTime.parse("1997-05-06 22:45:16", dateTimeFormatter); // parse = 1997-05-06T22:45:16 System.out.println("parse = " + parse); }
Instant類
在JDK8中給我們新增一個Instant類(時間戳/時間線),內(nèi)部保存了從1970年1月1日 00:00:00以來的秒和納秒。
@Test public void test01() throws Exception{ Instant now = Instant.now(); System.out.println("now = " + now); // 獲取從1970年一月一日 00:00:00 到現(xiàn)在的 納秒 System.out.println(now.getNano()); Thread.sleep(5); Instant now1 = Instant.now(); System.out.println("耗時:" + (now1.getNano() - now.getNano())); }
計算日期時間差
JDK8中提供了兩個工具類Duration/Period:計算日期時間差。
- Duration:用來計算兩個時間差(LocalTime)
- Period:用來計算兩個日期差(LocalDate)
@Test public void test01(){ // 計算時間差 LocalTime now = LocalTime.now(); LocalTime time = LocalTime.of(22, 48, 59); System.out.println("now = " + now); // 通過Duration來計算時間差 Duration duration = Duration.between(now, time); System.out.println(duration.toDays()); // 0 System.out.println(duration.toHours()); // 6 System.out.println(duration.toMinutes()); // 368 System.out.println(duration.toMillis()); // 22124240 // 計算日期差 LocalDate nowDate = LocalDate.now(); LocalDate date = LocalDate.of(1997, 12, 5); Period period = Period.between(date, nowDate); System.out.println(period.getYears()); // 23 System.out.println(period.getMonths()); // 5 System.out.println(period.getDays()); // 22 }
時間校正器
有時候我們可以需要如下調(diào)整:將日期調(diào)整到"下個月的第一天"等操作。這時我們通過時間校正器效果可能會更好。
- TemporalAdjuster:時間校正器
- TemporalAdjusters:通過該類靜態(tài)方法提供了大量的常用TemporalAdjuster的實現(xiàn)。
@Test public void test02(){ LocalDateTime now = LocalDateTime.now(); // 將當(dāng)前的日期調(diào)整到下個月的一號 TemporalAdjuster adJuster = (temporal)->{ LocalDateTime dateTime = (LocalDateTime) temporal; LocalDateTime nextMonth = dateTime.plusMonths(1).withDayOfMonth(1); System.out.println("nextMonth = " + nextMonth); return nextMonth; }; // 我們可以通過TemporalAdjusters 來實現(xiàn) // LocalDateTime nextMonth = now.with(adJuster); LocalDateTime nextMonth = now.with(TemporalAdjusters.firstDayOfNextMonth()); System.out.println("nextMonth = " + nextMonth); }
日期時間的時區(qū)
Java8 中加入了對時區(qū)的支持,LocalDate、LocalTime、LocalDateTime是不帶時區(qū)的,帶時區(qū)的日期時間類分別為:ZonedDate、ZonedTime、ZonedDateTime。其中每個時區(qū)都對應(yīng)著 ID,ID的格式為 “區(qū)域/城市” 。例如 :Asia/Shanghai 等。
ZoneId:該類中包含了所有的時區(qū)信息。
@Test public void test01(){ // 獲取所有的時區(qū)id // ZoneId.getAvailableZoneIds().forEach(System.out::println); // 獲取當(dāng)前時間 中國使用的 東八區(qū)的時區(qū),比標(biāo)準(zhǔn)時間早8個小時 LocalDateTime now = LocalDateTime.now(); System.out.println("now = " + now); // 2021-05-27T17:17:06.951 // 獲取標(biāo)準(zhǔn)時間 ZonedDateTime bz = ZonedDateTime.now(Clock.systemUTC()); System.out.println("bz = " + bz); // 2021-05-27T09:17:06.952Z // 使用計算機(jī)默認(rèn)的時區(qū),創(chuàng)建日期時間 ZonedDateTime now1 = ZonedDateTime.now(); System.out.println("now1 = " + now1); //2021-05-27T17:17:06.952+08:00[Asia/Shanghai] // 使用指定的時區(qū)創(chuàng)建日期時間 ZonedDateTime now2 = ZonedDateTime.now(ZoneId.of("America/Marigot")); System.out.println("now2 = " + now2); }
JDK新的日期和時間API的優(yōu)勢
- 新版日期時間API中,日期和時間對象是不可變,操作日期不會影響原來的值,而是生成一個新的實例
- 提供不同的兩種方式,有效的區(qū)分了人和機(jī)器的操作
- TemporalAdjuster可以更精確的操作日期,還可以自定義日期調(diào)整期
- 線程安全
以上就是Java8新特性O(shè)ptional類及新時間日期API示例詳解的詳細(xì)內(nèi)容,更多關(guān)于Java8 Optional類時間日期API的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
基于java+springboot+mybatis+laiyu實現(xiàn)學(xué)科競賽管理系統(tǒng)
這篇文章主要介紹了基于java+springboot+mybatis+laiyu實現(xiàn)的學(xué)科競賽管理系統(tǒng),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-09-09Java?OpenCV學(xué)習(xí)之Mat的基本操作詳解
OpenCV用來存儲圖像,很多時候都會用到這個Mat方法。數(shù)字圖像可看做一個數(shù)值矩陣,?其中的每一個元素表明一個像素點。Mat在?OpenCV?中表示的是?N?維稠密矩陣,與稠密矩陣相對的是稀疏矩陣。本文將重點介紹OpenCV中Mat的一些基本操作,需要的可以參考一下2022-03-03