Java8新特性Optional、接口中的默認方法與靜態方法


Optional

Optional 類(java.util.Optional) 是一個容器類,代表一個值存在或不存在,原來用 null 表示一個值不存在,現在 Optional 可以更好的表達這個概念。並且可以避免空指針異常。

常用方法:

Optional.of(T t) : 創建一個 Optional 實例。

Optional.empty() : 創建一個空的 Optional 實例。

Optional.ofNullable(T t):若 t 不為 null,創建 Optional 實例,否則創建空實例。

isPresent() : 判斷是否包含值。

orElse(T t) : 如果調用對象包含值,返回該值,否則返回t。

orElseGet(Supplier s) :如果調用對象包含值,返回該值,否則返回 s 獲取的值。

map(Function f): 如果有值對其處理,並返回處理后的Optional,否則返回 Optional.empty()。

flatMap(Function mapper):與 map 類似,要求返回值必須是Optional。

下面引用ImportNew的一段內容來告訴我們如何正確使用Optional。比如千萬不要寫成這樣子:

public static String getName(User u) {
    Optional<User> user = Optional.ofNullable(u);
    if (!user.isPresent()) return "Unknown"; return user.get().name; }

這樣改寫非但不簡潔,而且其操作還是和第一段代碼一樣。無非就是用isPresent方法來替代u==null。這樣的改寫並不是Optional正確的用法,我們再來改寫一次。

public static String getName(User u) { return Optional.ofNullable(u) .map(user->user.name) .orElse("Unknown"); }

這樣才是正確使用Optional的姿勢。那么按照這種思路,我們可以安心的進行鏈式調用,而不是一層層判斷了。看一段代碼:

public static String getChampionName(Competition comp) throws IllegalArgumentException { if (comp != null) { CompResult result = comp.getResult(); if (result != null) { User champion = result.getChampion(); if (champion != null) { return champion.getName(); } } } throw new IllegalArgumentException("The value of param comp isn't available."); }

由於種種原因(比如:比賽還沒有產生冠軍、方法的非正常調用、某個方法的實現里埋藏的大禮包等等),我們並不能開心的一路comp.getResult().getChampion().getName()到底。而其他語言比如kotlin,就提供了在語法層面的操作符加持:comp?.getResult()?.getChampion()?.getName()。所以講道理在Java里我們怎么辦!

讓我們看看經過Optional加持過后,這些代碼會變成什么樣子。

public static String getChampionName(Competition comp) throws IllegalArgumentException { return Optional.ofNullable(comp) .map(c->c.getResult()) .map(r->r.getChampion()) .map(u->u.getName()) .orElseThrow(()->new IllegalArgumentException("The value of param comp isn't available.")); }

這就很舒服了。Optional的魅力還不止於此,Optional還有一些神奇的用法,比如Optional可以用來檢驗參數的合法性。

public void setName(String name) throws IllegalArgumentException {
    this.name = Optional.ofNullable(name).filter(User::isNameValid) .orElseThrow(()->new IllegalArgumentException("Invalid username.")); }

上面代碼引用importnew—Java8 如何正確使用 Optional。

接口中的默認方法與靜態方法

Java8接口中可以添加靜態方法,也可以添加默認方法,默認方法用 default修飾。

public interface Fun<T> { default void getName(){ System.out.println("hello world"); } static void getAge(){ System.out.println("nine"); } }

若一個接口中定義了一個默認方法,他的實現類的一個父類定義了具有相同名稱和參數列表的方法。則調用該實現類的時候執行父類中的方法。

public class TestF { public void getName(){ System.out.println("TestF"); } } public interface TestInterface { default void getName(){ System.out.println("hello world"); } } public class Test extends TestF implements TestInterface{ public static void main(String[] args) { Test t = new Test(); t.getName();//輸出的是TestF } }

若一個實現類實現了兩個接口,如果一個父接口提供一個默認方法,而另一個父接口也提供了一個具有相同名稱和參數列表的方法(不管方法是否是默認方法),那么必須覆蓋該方法來解決沖突,否則會報錯。

public interface TestInterface { default void getName(){ System.err.println("hello world"); } } public interface TestInterface1 { void getName(); } public class Test1 implements TestInterface, TestInterface1{ public void getName(){ System.out.println("Tes1F"); } }
java學習群669823128


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM