JAVA 8 Optional類介紹及其源碼

来源:http://www.cnblogs.com/chenpi/archive/2016/10/08/5923829.html
-Advertisement-
Play Games

什麼是Optional對象 Java 8中所謂的Optional對象,即一個容器對象,該對象可以包含一個null或非null值。如果該值不為null,則調用isPresent()方法將返回true,且調用get()方法會返回該值。 另外,該對象還有其它方法: 如可以使用orElse()方法給Opti ...


什麼是Optional對象

Java 8中所謂的Optional對象,即一個容器對象,該對象可以包含一個null或非null值。如果該值不為null,則調用isPresent()方法將返回true,且調用get()方法會返回該值。

另外,該對象還有其它方法:

如可以使用orElse()方法給Optional對象設定預設值(當值為null時,會使用預設值);

使用ifPresent()方法來執行一段代碼(當值不為null時,執行代碼段)。

Optional主要被用於Java 8的Stream中,簡單舉個例子:

package optional;

import java.util.Optional;
import java.util.stream.Stream;

public class Snippet
{
    public static void main(String[] args)
    {
        Stream<String> names = Stream.of("Lamurudu", "Okanbi", "Oduduwa");
        
        Optional<String> startswl = names.filter(name -> name.startsWith("L")).findFirst();
        
        //判斷是否不為null
        if(startswl.isPresent()){
            System.out.println(startswl.get());
        }
        
        //if值為null:列印“null”;if值不為null:列印原值
        System.out.println(startswl.orElse("null"));

        //if值不為null,執行Lambda表達式
        startswl.ifPresent(name -> {
            String s = name.toUpperCase();
            System.out.println(s);
        });
    }
}

使用Optional對象的好處

減少NullPointerException異常

寫出更加優雅的代碼

源碼及示例

Optional類的屬性和方法如下:

 

我們一個個看,先看兩個成員屬性;

成員屬性

如下,一個是EMPTY常量,即存放空值的Optional對象,另一個是value,即被存放的值,可為null或非null值;

    /**
     * Common instance for {@code empty()}.
     */
    private static final Optional<?> EMPTY = new Optional<>();

    /**
     * If non-null, the value; if null, indicates no value is present
     */
    private final T value;

構造方法

兩個構造方法,註意都是私有的

1、創建一個包含空值的Optional對象;

2、創建一個非空值的Optional對象;

    private Optional() {
        this.value = null;
    }
    private Optional(T value) {
        this.value = Objects.requireNonNull(value);
    }

empty()方法

這個方法很簡單,作用是返回一個Optional實例,裡面存放的value是null,源碼如下:

    public static<T> Optional<T> empty() {
        @SuppressWarnings("unchecked")
        Optional<T> t = (Optional<T>) EMPTY;
        return t;
    }

of(T value)方法

很簡單,就是返回一個包含非空值的Optional對象

    public static <T> Optional<T> of(T value) {
        return new Optional<>(value);
    }

ofNullable(T value)方法

 很簡單,返回一個可以包含空值的Optional對象

    public static <T> Optional<T> ofNullable(T value) {
        return value == null ? empty() : of(value);
    }

get()方法

 得到Optional對象里的值,如果值為null,則拋出NoSuchElementException異常

    public T get() {
        if (value == null) {
            throw new NoSuchElementException("No value present");
        }
        return value;
    }

isPresent()方法

很簡單,判斷值是否不為null

    public boolean isPresent() {
        return value != null;
    }

ifPresent(Consumer<? super T> consumer)方法

 當值不為null時,執行consumer

    public void ifPresent(Consumer<? super T> consumer) {
        if (value != null)
            consumer.accept(value);
    }

舉個例子,ifPresent方法執行Lambda表達式,將值轉換為大寫並列印:

package optional;

import java.util.Optional;

public class Snippet
{
    public static void main(String[] args)
    {
        Optional<String> test = Optional.ofNullable("abcDef");
        
        //值不為null,執行Lambda表達式,
        test.ifPresent(name -> {
            String s = name.toUpperCase();
            System.out.println(s);
        });
        //列印ABCDEF
    }
}

filter(Predicate<? super T> predicate)方法

看方法名就知道,該方法是過濾方法,過濾符合條件的Optional對象,這裡的條件用Lambda表達式來定義,

如果入參predicate對象為null將拋NullPointerException異常,

如果Optional對象的值為null,將直接返回該Optional對象,

如果Optional對象的值符合限定條件(Lambda表達式來定義),返回該值,否則返回空的Optional對象

源碼如下:

    public Optional<T> filter(Predicate<? super T> predicate) {
        Objects.requireNonNull(predicate);
        if (!isPresent())
            return this;
        else
            return predicate.test(value) ? this : empty();
    }

使用示例:

package optional;

import java.util.Optional;

public class Snippet
{
    public static void main(String[] args)
    {
        Optional<String> test = Optional.ofNullable("abcD");
        
        //過濾值的長度小於3的Optional對象
        Optional<String> less3 = test.filter((value) -> value.length() < 3);
        //列印結果
        System.out.println(less3.orElse("不符合條件,不列印值!"));
    }
}

map(Function<? super T, ? extends U> mapper)方法

 前面的filter方法主要用於過濾,一般不會修改Optional裡面的值,map方法則一般用於修改該值,並返回修改後的Optional對象

如果入參mapper對象為null將拋NullPointerException異常,

如果Optional對象的值為null,將直接返回該Optional對象,

最後,執行傳入的lambda表達式,並返回經lambda表達式操作後的Optional對象

    public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {
        Objects.requireNonNull(mapper);
        if (!isPresent())
            return empty();
        else {
            return Optional.ofNullable(mapper.apply(value));
        }
    }

使用示例:

package optional;

import java.util.Optional;

public class Snippet
{
    public static void main(String[] args)
    {
        Optional<String> test = Optional.ofNullable("abcD");
        
        //將值修改為大寫
        Optional<String> less3 = test.map((value) -> value.toUpperCase());
        //列印結果 ABCD
        System.out.println(less3.orElse("值為null,不列印!"));
    }
}

flatMap(Function<? super T, Optional<U>> mapper)方法

flatMap方法與map方法基本一致,唯一的區別是,

如果使用flatMap方法,需要自己在Lambda表達式里將返回值轉換成Optional對象,

而使用map方法則不需要這個步驟,因為map方法的源碼里已經調用了Optional.ofNullable方法;

源碼:

    public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {
        Objects.requireNonNull(mapper);
        if (!isPresent())
            return empty();
        else {
            return Objects.requireNonNull(mapper.apply(value));
        }
    }

使用示例:

package optional;

import java.util.Optional;

public class Snippet
{
    public static void main(String[] args)
    {
        Optional<String> test = Optional.ofNullable("abcD");
        
        //使用flatMap,將值修改為大寫
        Optional<String> less3 = test.flatMap((value) -> Optional.ofNullable(value.toUpperCase()));
        //使用map,將值修改為大寫
        //Optional<String> less3 = test.map((value) -> value.toUpperCase());
        
        //列印結果 ABCD
        System.out.println(less3.orElse("值為null,不列印!"));
    }
}

orElse(T other)方法

很簡單,當值為null時返回傳入的值,否則返回原值;

源碼:

    public T orElse(T other) {
        return value != null ? value : other;
    }

orElseGet(Supplier<? extends T> other)方法

功能與orElse(T other)類似,不過該方法可選值的獲取不是通過參數直接獲取,而是通過調用傳入的Lambda表達式獲取

源碼:

    public T orElseGet(Supplier<? extends T> other) {
        return value != null ? value : other.get();
    }

使用示例:

package optional;

import java.util.Optional;

public class Snippet
{
    public static void main(String[] args)
    {
        Optional<String> test = Optional.ofNullable(null);

        System.out.println(test.orElseGet(() -> "hello"));
        //將列印hello
    }
}

orElseThrow(Supplier<? extends X> exceptionSupplier)方法

當遇到值為null時,根據傳入的Lambda表達式跑出指定異常

源碼

    public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
        if (value != null) {
            return value;
        } else {
            throw exceptionSupplier.get();
        }
    }

使用示例:

package optional;

import java.util.Optional;

public class Snippet
{
    public static void main(String[] args)
    {
        Optional<String> test = Optional.ofNullable(null);

        //這裡的Lambda表達式為構造方法引用
        System.out.println(test.orElseThrow(NullPointerException::new));
        //將列印hello
    }
}

參考資料

jdk1.8.0_31源碼

https://blog.idrsolutions.com/2015/04/java-8-optional-class-explained-in-5-minutes/

https://www.voxxed.com/blog/2015/05/why-even-use-java-8-optional/

http://unmi.cc/proper-ways-of-using-java8-optional/


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

-Advertisement-
Play Games
更多相關文章
  • ASP.NET Core MVC 控制器應通過它們的構造器明確的請求它們的依賴關係。在某些情況下,單個控制器的操作可能需要一個服務,在控制器級別上的請求可能沒有意義。在這種情況下,你也可以選擇將服務作為 action 方法的參數。 ...
  • VS "15" 預覽 5 給 VB 帶來了更新。 * 值元組 ValueTuple * 二進位數字和數字分隔符 * 支持使用 ByRef 返回類型 ...
  • ASP.NET MVC 過濾器 可在執行管道的前後特定階段執行代碼。過濾器可以配置為全局有效、僅對控制器有效或是僅對 Action 有效。 ...
  • 一、綜述 SuperIO(SIO)定位在PC終端(上位機)應用,它只有一個服務實例,配置參數是全局屬性。但是,ServerSuperIO(SSIO)與SuperIO(SIO)定位不一樣,SSIO定位在伺服器端,不管是串口通訊模式,還是網路通訊模式,都支持多服務實例,所以每個服務實例都有自己的配置參數 ...
  • 今天寫了一個英漢詞典小程式,我加了好多註釋,適合初學者一起參考,哪裡寫的不好請幫忙指出,一起學習進步。。 這裡用到了,泛型,泛型字典,一些控制項的操作,split的應用,數組的應用,時間間隔,linkLabel的使用。。 using System; using System.Collections.G ...
  • 註明:相關資料參考來自網路 PHP中使用fopen()函數打開文件,函數語法fopen(filename,mode,include_path,context),filename為要打開的文件,mode為訪問類型,include_path和context為可選。 mode說明 r 只讀方式打開,在文件 ...
  • 【轉自】http://blog.csdn.net/foamflower/article/details/5713604 isNotEmpty將空格也作為參數,isNotBlank則排除空格參數 StringUtils方法的操作對象是java.lang.String類型的對象,是JDK提供的Strin ...
  • 在網路編程中,出於節約帶寬或者編碼的需要,通常需要以原生方式處理long和int,而不是轉換為string。 public class ByteOrderUtils { public static byte[] int2byte(int res) { byte[] targets = new byt ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...