成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

給女朋友講解什么是Optional【JDK 8特性】

caspar / 3319人閱讀

摘要:接口例子如果容器的對象存在,則對其執(zhí)行調(diào)用函數(shù)得到返回值。上面一句代碼對應(yīng)著最開始的老寫法方法直接看源碼方法與方法類似,區(qū)別在于函數(shù)的返回值不同。

前言
只有光頭才能變強

前兩天帶女朋友去圖書館了,隨手就給她來了一本《與孩子一起學編程》的書,于是今天就給女朋友講解一下什么是Optional類。

至于她能不能看懂,那肯定是看不懂的。(學到變量/for循環(huán)的女人怎么能看懂呢)

不知道大家還記得上一篇《阿里巴巴 Java開發(fā)手冊》讀后感不,當時閱讀到空指針異常(NPE)時,書上提到JDK 8有個Optional類供我們使用,該類可以盡可能地防止出現(xiàn)空指針異常(NPE)。

文本力求簡單講清每個知識點,希望大家看完能有所收獲

一、基礎(chǔ)鋪墊

我們都知道JDK 8最重要的新特性是Lambda表達式,這個可以讓我們簡化非常多的代碼編寫,不知道大家會使用了沒有。這里我簡單跟大家來回顧一下~

1.1Lambda簡化代碼例子

下面就以幾個例子來看看Lambda表達式是怎么簡化我們代碼的編寫的。

首先我們來看看創(chuàng)建線程

public static void main(String[] args) {
    // 用匿名內(nèi)部類的方式來創(chuàng)建線程
    new Thread(new Runnable() {
        @Override
        public void run() {
            System.out.println("公眾號:Java3y---回復(fù)1進群交流");
        }
    });

    // 使用Lambda來創(chuàng)建線程
    new Thread(() -> System.out.println("公眾號:Java3y---回復(fù)1進群交流"));
}

再來看看遍歷Map集合:


public static void main(String[] args) {
    Map hashMap = new HashMap<>();
    hashMap.put("公眾號", "Java3y");
    hashMap.put("交流群", "回復(fù)1");

    // 使用增強for的方式來遍歷hashMap
    for (Map.Entry entry : hashMap.entrySet()) {
        System.out.println(entry.getKey()+":"+entry.getValue());
    }

    // 使用Lambda表達式的方式來遍歷hashMap
    hashMap.forEach((s, s2) -> System.out.println(s + ":" + s2));
}

在List中刪除某個元素

public static void main(String[] args) {

    List list = new ArrayList<>();
    list.add("Java3y");
    list.add("3y");
    list.add("光頭");
    list.add("帥哥");
    
    // 傳統(tǒng)的方式刪除"光頭"的元素
    ListIterator iterator = list.listIterator();
    while (iterator.hasNext()) {
        if ("光頭".equals(iterator.next())) {
            iterator.remove();
        }
    }

    // Lambda方式刪除"光頭"的元素
    list.removeIf(s -> "光頭".equals(s));
    
    // 使用Lambda遍歷List集合
    list.forEach(s -> System.out.println(s));
}

從上面的例子我們可以看出,Lambda表達式的確是可以幫我們簡化代碼的。

1.1函數(shù)式接口

使用Lambda表達式,其實都是建立在函數(shù)式接口上的。我們看看上面的代碼的接口:

創(chuàng)建多線程的Runnable接口:

@FunctionalInterface
public interface Runnable {
    public abstract void run();
}

遍歷HashMap的BiConsumer接口:

@FunctionalInterface
public interface BiConsumer {
    void accept(T t, U u);
    default BiConsumer andThen(BiConsumer after) {
        Objects.requireNonNull(after);
        return (l, r) -> {
            accept(l, r);
            after.accept(l, r);
        };
    }
}

在List中刪除元素的Predicate接口:

@FunctionalInterface
public interface Predicate {

    boolean test(T t);

    default Predicate and(Predicate other) {
        Objects.requireNonNull(other);
        return (t) -> test(t) && other.test(t);
    }
    default Predicate negate() {
        return (t) -> !test(t);
    }
    default Predicate or(Predicate other) {
        Objects.requireNonNull(other);
        return (t) -> test(t) || other.test(t);
    }
    static  Predicate isEqual(Object targetRef) {
        return (null == targetRef)
                ? Objects::isNull
                : object -> targetRef.equals(object);
    }
}

函數(shù)式接口的特點:由@FunctionalInterface注解標識,接口有且僅有一個抽象方法!

1.2Lambda簡單講解

或許我們一開始看到Lambda的時候,發(fā)現(xiàn)Lambda表達式的語法有點奇葩,甚至有點看不懂。沒事,這里3y給大家用圖的形式畫一畫:

以Runnable接口來舉例:

再不濟,我們在用IDE的時候,可以提示出Lambda表達式的語法的,這樣可以幫我們快速上手Lambda表達式:

說白了,我們使用Lambda表達式的架子是這樣的()->{},具體的時候看看函數(shù)式接口的抽象方法要求就可以了,再不濟就使用IDE智能提示。

1.3泛型回顧

比如說public Optional map(Function mapper)這個聲明,你看懂了嗎?

// 接口
@FunctionalInterface
public interface Function {
    R apply(T t);
}

在泛型的上限和下限中有一個原則:PECS(Producer Extends Consumer Super)

帶有子類限定的可以從泛型讀取【也就是--->(? extend T)】-------->Producer Extends

帶有超類限定的可以從泛型寫入【也就是--->(? super T)】-------->Consumer Super

解析:傳入的參數(shù)是泛型 T 或者其父類,返回值是U或其子類。

具體可參考:

泛型就這么簡單

二、Optional類

一句話介紹Optional類:使用JDK8的Optional類來防止NPE(空指針異常)問題。

接下來我們看看文檔是怎么說的:

A container object which may or may not contain a non-null value.Additional methods that depend on the presence or absence of a contained value are provided

它是一個容器,裝載著非NULL元素(或者沒有裝載元素),提供了一系列的方法供我們判斷該容器里的對象是否存在(以及后續(xù)的操作)。

Optional類的方法結(jié)構(gòu)圖:

2.1創(chuàng)建Optional容器

我們先來看看Optional的屬性以及創(chuàng)建Optional容器的方法:

    // 1、創(chuàng)建出一個Optional容器,容器里邊并沒有裝載著對象
    private static final Optional EMPTY = new Optional<>();

    // 2、代表著容器中的對象
    private final T value;

    // 3、私有構(gòu)造方法
    private Optional() {
        this.value = null;
    }

    // 4、得到一個Optional容器,Optional沒有裝載著對象
    public static Optional empty() {
        @SuppressWarnings("unchecked")
        Optional t = (Optional) EMPTY;
        return t;
    }

    // 5、私有構(gòu)造方法(帶參數(shù)),參數(shù)就是具體的要裝載的對象,如果傳進來的對象為null,拋出異常
    private Optional(T value) {
        this.value = Objects.requireNonNull(value);
    }

    // 5.1、如果傳進來的對象為null,拋出異常
    public static  T requireNonNull(T obj) {
        if (obj == null)
            throw new NullPointerException();
        return obj;
    }


    // 6、創(chuàng)建出Optional容器,并將對象(value)裝載到Optional容器中。
    // 傳入的value如果為null,拋出異常(調(diào)用的是Optional(T value)方法)
    public static  Optional of(T value) {
        return new Optional<>(value);
    }

    // 創(chuàng)建出Optional容器,并將對象(value)裝載到Optional容器中。
    // 傳入的value可以為null,如果為null,返回一個沒有裝載對象的Optional對象
    public static  Optional ofNullable(T value) {
        return value == null ? empty() : of(value);
    }

所以可以得出創(chuàng)建Optional容器有兩種方式:

調(diào)用ofNullable()方法,傳入的對象可以為null

調(diào)用of()方法,傳入的對象不可以為null,否則拋出NullPointerException

下面我們簡單就可以看看用法了:

現(xiàn)在我有一個User對象,這里用到了Lombok,有興趣的同學可去學學了解一下:兩個月的Java實習結(jié)束,繼續(xù)努力

import lombok.Data;
@Data
public class User {

    private Integer id;
    private String name;
    private Short age;
}

測試:

public static void main(String[] args) {

    User user = new User();
    User user1 = null;

    // 傳遞進去的對象不可以為null,如果為null則拋出異常
    Optional op1 = Optional.of(user1);

    // 傳遞進去的對象可以為null,如果為null則返回一個沒有裝載對象的Optional容器
    Optional op2 = Optional.ofNullable(user);
}

2.2Optional容器簡單的方法
// 得到容器中的對象,如果為null就拋出異常
public T get() {
    if (value == null) {
        throw new NoSuchElementException("No value present");
    }
    return value;
}

// 判斷容器中的對象是否為null
public boolean isPresent() {
    return value != null;
}

// 如果容器中的對象存在,則返回。否則返回傳遞進來的參數(shù)
public T orElse(T other) {
    return value != null ? value : other;
}

這三個方法是Optional類比較常用的方法,并且是最簡單的。(因為參數(shù)不是函數(shù)式接口)

下面我們繼續(xù)看看用法:

public static void main(String[] args) {

        User user = new User();
        User user1 = null;

        Optional op1 = Optional.ofNullable(user);
        System.out.println(op1.isPresent());
        System.out.println(op1.get());
        System.out.println(op1.orElse(user1));

    }

結(jié)果很明顯,因為我們的user是不為null的:

我們調(diào)換一下順序看看:

public static void main(String[] args) {

    User user = new User();
    User user1 = null;

    Optional op1 = Optional.ofNullable(user1);
    System.out.println(op1.isPresent());
    System.out.println(op1.orElse(user));
    System.out.println(op1.get());

}

2.3Optional容器進階用法

當然了,我們到目前為止看起來Optional類好像就這么一回事了,這樣代碼寫起來還不如我自己判斷null呢...

我們對比一下:

我們可以發(fā)現(xiàn),手動判斷是否為null好像還更方便簡潔一點呢。

所以,我們帶函數(shù)式接口的方法登場了!

2.3.1ifPresent方法

首先來看看ifPresent(Consumer consumer)方法


public void ifPresent(Consumer consumer) {
    if (value != null)
        consumer.accept(value);
}

@FunctionalInterface
public interface Consumer {
    void accept(T t);
}

如果容器中的對象存在,則調(diào)用accept方法,比如說:

public static void main(String[] args) {

    User user = new User();
    user.setName("Java3y");
    test(user);
}

public static void test(User user) {

    Optional optional = Optional.ofNullable(user);

    // 如果存在user,則打印user的name
    optional.ifPresent((value) -> System.out.println(value.getName()));

    // 舊寫法
    if (user != null) {
        System.out.println(user.getName());
    }
}
2.3.2orElseGet和orElseThrow方法

直接看源碼:

// 如果對象存在,則直接返回,否則返回由Supplier接口的實現(xiàn)用來生成默認值
public T orElseGet(Supplier other) {
    return value != null ? value : other.get();
}


@FunctionalInterface
public interface Supplier {
    T get();
}


// 如果存在,則返回。否則拋出supplier接口創(chuàng)建的異常
public  T orElseThrow(Supplier exceptionSupplier) throws X {
    if (value != null) {
        return value;
    } else {
        throw exceptionSupplier.get();
    }
}

例子:

public static void main(String[] args) {

    User user = new User();
    user.setName("Java3y");
    test(user);
}

public static void test(User user) {

    Optional optional = Optional.ofNullable(user);

    // 如果存在user,則直接返回,否則創(chuàng)建出一個新的User對象
    User user1 = optional.orElseGet(() -> new User());
    
    // 舊寫法
    if (user != null) {
        user = new User();
    }
}

總的來說跟我們上面所講的orElse()差不多,只不過它可以通過Supplier接口的實現(xiàn)來生成默認值。

2.3.3filter方法

直接看源碼:

// 如果容器中的對象存在,并且符合過濾條件,返回裝載對象的Optional容器,否則返回一個空的Optional容器
public Optional filter(Predicate predicate) {
    Objects.requireNonNull(predicate);
    if (!isPresent())
        return this;
    else
        return predicate.test(value) ? this : empty();
}


// 接口
@FunctionalInterface
public interface Predicate {

    boolean test(T t);
}

返回Optional對象我們就可以實現(xiàn)鏈式調(diào)用了!

例子:

public static void test(User user) {

    Optional optional = Optional.ofNullable(user);

    // 如果容器中的對象存在,并且符合過濾條件,返回裝載對象的Optional容器,否則返回一個空的Optional容器
    optional.filter((value) -> "Java3y".equals(value.getName()));
}
2.3.4map方法

直接看源碼:

// 如果容器的對象存在,則對其執(zhí)行調(diào)用mapping函數(shù)得到返回值。然后創(chuàng)建包含mapping返回值的Optional,否則返回空Optional。
public Optional map(Function mapper) {
    Objects.requireNonNull(mapper);
    if (!isPresent())
        return empty();
    else {
        return Optional.ofNullable(mapper.apply(value));
    }
}


// 接口
@FunctionalInterface
public interface Function {
    R apply(T t);
}

例子:

public static void test(User user) {

    Optional optional = Optional.ofNullable(user);

    // 如果容器的對象存在,則對其執(zhí)行調(diào)用mapping函數(shù)得到返回值。然后創(chuàng)建包含mapping返回值的Optional,否則返回空Optional。
    optional.map(user1 -> user1.getName()).orElse("Unknown");
}

// 上面一句代碼對應(yīng)著最開始的老寫法:

public String tradition(User user) {
    if (user != null) {
        return user.getName();
    }else{
        return "Unknown";
    }
}
2.3.5flatMap方法

直接看源碼:

// flatMap方法與map方法類似,區(qū)別在于apply函數(shù)的返回值不同。map方法的apply函數(shù)返回值是? extends U,而flatMap方法的apply函數(shù)返回值必須是Optional
public Optional flatMap(Function> mapper) {
    Objects.requireNonNull(mapper);
    if (!isPresent())
        return empty();
    else {
        return Objects.requireNonNull(mapper.apply(value));
    }
}
2.3.6總結(jié)

再來感受一下Optional的魅力

public static void main(String[] args) {
    User user = new User();
    user.setName("Java3y");
    System.out.println(test(user));
}

// 以前的代碼v1
public static String test2(User user) {
    if (user != null) {
        String name = user.getName();
        if (name != null) {
            return name.toUpperCase();
        } else {
            return null;
        }
    } else {
        return null;
    }
}

// 以前的代碼v2
public static String test3(User user) {
    if (user != null && user.getName() != null) {
        return user.getName().toUpperCase();
    } else {
        return null;
    }
}

// 現(xiàn)在的代碼
public static String test(User user) {
    return Optional.ofNullable(user)
            .map(user1 -> user1.getName())
            .map(s -> s.toUpperCase()).orElse(null);
}

Optional總結(jié):

filter,map或flatMap一個函數(shù),函數(shù)的參數(shù)拿到的值一定不是null。所以我們通過filter,map 和 flatMap之類的函數(shù)可以將其安全的進行變換,最后通過orElse系列,get,isPresent 和 ifPresent將其中的值提取出來。

其實吧,用Optional類也沒有簡化很多的代碼,只是把NPE異常通過各種方法隱藏起來(包裝了一層)。通過Lambda表達式可以讓我們處理起來更加"優(yōu)雅"一些。

三、最后

之前在初學的時候沒在意JDK8的特性,其實JDK更新很多時候都能給我們帶來不少好處的(簡化代碼編寫,提高性能等等),所以作為一名Java程序員,還是得多學學新特性。(話說JDK9該類又有新特性了...)

如果你要評論“醒醒吧,程序員哪來的女朋友”,“我尿黃,讓我來”之類的話,我建議你是不是好好反省一下自己,為什么別的程序員都有女朋友,就你沒有,是不是自己技術(shù)不過關(guān)了?通過“工廠”找一個有那么難嗎?再不濟也能自己new一個出來啊。

當然了,我的女朋友是現(xiàn)實存在的。

參考資料:

Java 8 Optional類深度解析:https://www.cnblogs.com/xingzc/p/5778090.html

Java8 如何正確使用 Optional:http://www.importnew.com/26066.html

https://www.zhihu.com/question/63783295/answer/214531004

【Java】jdk8 Optional 的正確姿勢https://blog.csdn.net/hj7jay/article/details/52459334

如果你覺得我寫得還不錯,了解一下:

堅持原創(chuàng)的技術(shù)公眾號:Java3y。回復(fù) 1 加入Java交流群

文章的目錄導航(精美腦圖+海量視頻資源):https://github.com/ZhongFuCheng3y/3y

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/72280.html

相關(guān)文章

  • Java核心技術(shù)教程整理,長期更新

    以下是Java技術(shù)棧微信公眾號發(fā)布的關(guān)于 Java 的技術(shù)干貨,從以下幾個方面匯總。 Java 基礎(chǔ)篇 Java 集合篇 Java 多線程篇 Java JVM篇 Java 進階篇 Java 新特性篇 Java 工具篇 Java 書籍篇 Java基礎(chǔ)篇 8張圖帶你輕松溫習 Java 知識 Java父類強制轉(zhuǎn)換子類原則 一張圖搞清楚 Java 異常機制 通用唯一標識碼UUID的介紹及使用 字符串...

    Anchorer 評論0 收藏0
  • 聊聊 Java8 以后各個版本的新特性

    摘要:于是抽時間看了看以后各個版本的特性,做了一個總結(jié)。年和公開版本發(fā)布,取名為。此后對應(yīng)版本就是,。發(fā)布,是一個重大版本更新。在此之后,就是每六個月發(fā)布一次新版本。以上和參考資料聊了一些關(guān)于的歷史,下面我們看看各個版本有那些新特性。 【這是 ZY 第 11 篇原創(chuàng)技術(shù)文章】 某天在網(wǎng)上閑逛,突然看到有篇介紹 Java 11 新特性的文章,頓時心里一驚,畢竟我對于 Java 的版本認識...

    K_B_Z 評論0 收藏0
  • JDK 10 的新特性和增強功能

    摘要:的問題在于,版本號中編碼了它和它對之前版本的兼容性信息。但是在六個月節(jié)奏的情況下,這些信息都是未知的,在發(fā)布前任何事情都可能發(fā)生,由此規(guī)范下的版本號也會是未知的。程序會對文件的完整性做一個保護,因此修改既可能丟失。 本文是對底部參考資料的整理得到的,由于本人技術(shù)水平和英語水平都不是很高,有些詞如有翻譯錯誤或句子的理解錯誤還請指出。 JEP 286 局部變量推斷: var 傳統(tǒng)的 J...

    yibinnn 評論0 收藏0
  • Jdk1.8特性學習(Optional

    摘要:它的出現(xiàn)是為我們解決空指針異常的,以前我們寫代碼如果不進行判斷,會經(jīng)常出現(xiàn)異常。因為它本身就是個對象,不管放進去的對象為不為,始終不會返回,所以你也不需要在你的業(yè)務(wù)流程中進行一大堆的判斷,避免了程序運行時的空指針異常。 想必大家已經(jīng)在使用jdk1.8做項目開發(fā),但是你對于它里面的一些性特性了解多少呢?有沒有在你的項目中運用呢?現(xiàn)在就和我來一起梳理一下吧。 介紹 它是java.util包...

    liaosilzu2007 評論0 收藏0
  • 樂字節(jié)-Java8特性-接口默認方法

    摘要:注意當多個父接口中存在相同的默認方法時,子類中以就近原則繼承。定義靜態(tài)默認方法這是版簡易計算器接口默認方法使用定義接口并提供默認打印方法定義接口默認方法支持方法形參這是數(shù)值運算基本接口。。。 總概 JAVA8 已經(jīng)發(fā)布很久,而且毫無疑問,java8是自java5(2004年發(fā)布)之后的最重要的版本。其中包括語言、編譯器、庫、工具和JVM等諸多方面的新特性。 Java8 新特性列表如下:...

    arashicage 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<