成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

代碼優雅之道——消滅空指針,Optional來幫忙!

開發 前端
最近在看《Java開發手冊》,一直想著提高自己的代碼水平,文中就指出了使用Optional來解決NullPointerException!

一、前言

我們在開發中最常見的異常就是NullPointerException,防不勝防啊,相信大家肯定被坑過!

這種基本出現在獲取數據庫信息中、三方接口,獲取的對象為空,再去get出現!

解決方案當然簡單,只需要判斷一下,不是空在去后續操作,為空返回!

所有在JDK8時出現了專門處理的方案,出來很早了,但是小編慚愧一直沒有去使用它!

最近在看《Java開發手冊》,一直想著提高自己的代碼水平,文中就指出了使用Optional來解決NullPointerException!

二、Java開發手冊規范

小編使用的是2022版的黃山版,29頁寫到:

【推薦】防止 NPE,是程序員的基本修養,注意 NPE 產生的場景:

  • 返回類型為基本數據類型,return 包裝數據類型的對象時,自動拆箱有可能產生 NPE

反例:public int method() { return Integer 對象; },如果為 null,自動解箱拋 NPE。

  • 數據庫的查詢結果可能為 null。
  • 集合里的元素即使 isNotEmpty,取出的數據元素也可能為 null。
  • 遠程調用返回對象時,一律要求進行空指針判斷,防止 NPE。
  • 對于 Session 中獲取的數據,建議進行 NPE 檢查,避免空指針。
  • 級聯調用 obj.getA().getB().getC();一連串調用,易產生 NPE。

正例:使用 JDK8 的 Optional 類來防止 NPE 問題。

這份手冊還是不錯的,推薦反復閱讀,雖然進不去大廠,也要自覺約束自己的代碼風格,努力向大廠靠!

大家現在不知道哪里找的可以下載一下:

《Java開發手冊》:https://github.com/alibaba/p3c

三、Optional常用方法

小編帶大家一起從api文檔中的方法,一個個帶大家慢慢去了解它!

1、empty()

返回一個空的Optional實例:Optional.empty

Optional<Object> empty = Optional.empty();
log.info("empty值:{}",empty);

2、of(T value)

傳入一個參數,返回一個Optional對象,如果參數為空,報NullPointerException!

Test testNew  = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional對象:{}",optionalNew);
Optional<Test> optional = Optional.of(test);

源碼查看:

我們看到參數為空會報NullPointerException,我們去方法內部看一下就明白了:

public static <T> Optional<T> of(T value) {
    return new Optional<>(value);
}
private Optional(T value) {
    this.value = Objects.requireNonNull(value);
}
public static <T> T requireNonNull(T obj) {
    if (obj == null)
        throw new NullPointerException();
    return obj;
}

我們發現是在Objects類中的requireNonNull方法中判斷了是否為空!

這個還會出現NullPointerException,所以我們一般使用下面的這個方法!

3、ofNullable(T value)

參數傳入一個對象,返回一個Optional對象,如果為空,將返回一個空的Optional對象,就等于Optional.empty。

Test testNew  = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional對象:{}",optionalNew);

Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional對象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional對象中的ofNullable方法new返回值:{}",optionalTestNew);

源碼查看:

我們發現是在方法開始進行非空判斷,再去調用上面的of(T value)方法。

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

4、get()

如果此Optional中存在值,則返回該值,否則拋出NoSuchElementException。

Test testNew  = new Test();
Test test = null;
Optional<Test> optionalNew = Optional.of(testNew);
log.info(" optional對象:{}",optionalNew);
// Optional<Test> optional = Optional.of(test);

Optional<Test> optionalTest = Optional.ofNullable(test);
log.info(" optional對象中的ofNullable方法返回值:{}",optionalTest);
Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
log.info(" optional對象中的ofNullable方法new返回值:{}",optionalTestNew);

Test test2 = optionalTestNew.get();
log.info("原來有值的:經過Optional包裝后get后得到原來的值:{}",test2);
Test test1 = optionalTest.get();
log.info("原來沒有值的:經過Optional包裝后get后得到原來的值:{}",test1);

源碼查看:

調用開始會進行值判斷,如果為空則拋異常!

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

5、isPresent()

如果存在值,則返回true,否則返回false。

這里代碼就不加上面的,大家參考上面的獲取一個Optional對象。

boolean present = optionalTestNew.isPresent();
log.info("optionalTestNew調用是否為空:{}",present);
boolean present1 = optionalTest.isPresent();
log.info("optionalTest調用是否為空:{}",present1);

源碼查看:

這就比較簡單了!

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

6、ifPresent(Consumer<? super T> consumer)

如果存在值,則使用該值調用指定的使用者,否則不執行任何操作。

主要的就是入參數一個函數式接口,有值就會去執行,為空則不進行任何操作!

小技巧:

開始對lambda不了解時,可以先按照上面這種方式進行寫,

大家可以看到Idea給置灰了,就是可以優化,我們Alt+Enter然后再次Enter就會變成后面的lambda!

optionalTest.ifPresent(new Consumer<Test>() {
    @Override
    public void accept(Test test) {
        log.info("我是調用ifPresent執行后的打印=====");
    }
});
optionalTestNew.ifPresent(testInner -> log.info("我是調用ifPresent執行后的打印"));

源碼查看:

還是先判斷不為空才去執行函數式接口!

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

7、filter(Predicate<? super T> predicate)

如果存在值,并且該值符合規則,則返回描述該值的Optional,否則返回空Optional

是一個Predicate函數接口,可以傳入實現了Predicate接口的lambda表達式! 如果不符合條件就會返回一個Optional.empty。

testNew.setName("蕭炎");
testNew.setAge(33);
Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);
log.info("過濾后的結果:{}",optionalTest1.get());

源碼查看:

就是判斷一下表達式和值是否為空,然后就是根據規則判斷。

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

8、map(Function<? super T,? extends U> mapper)

如果存在值,則將提供的映射函數應用于該值,如果結果為非空,則返回描述結果的Optional。否則,返回空的Optional。

也是一個函數式接口!

Optional<String> stringOptional = optionalTestNew.map(Test::getName);
log.info("map后獲得字段值:{}",stringOptional.get());

源碼查看:

也是進行非空判斷,然后執行lambda得到字段后放到ofNullable方法中!

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));
    }
}

9、flatMap(Function<? super T,Optional<U>> mapper)

如果存在值,則將提供的Optional方位映射函數應用于該值,返回該結果,否則返回空的Optional。此方法類似于map,但提供的映射器的結果已經是可選的,并且如果調用,flatMap不會不會在最后進行任何包裝。

Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);
log.info("flatMap后得到的字段:{}",optional.get());

private static Optional<String> getFlatMap(Test test){
    return Optional.ofNullable(test).map(Test::getName);
}

源碼查看:

也是進行非空判斷,然后和map不同的是不執行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));
    }
}

10、orElse(T other)

如果有值則將其返回,否則返回指定的其它值。

如果你是一個對象,orElse()也要是相同對象!

String message = null;
String messageNew = "關注公眾號:小王博客基地";

String nullString = Optional.ofNullable(message).orElse("這是一個空字符串!");
log.info("這是空字符串打印的:{}",nullString);
String string = Optional.ofNullable(messageNew).orElse("=====這是一個空字符串!");
log.info("這是字符串打印的:{}",string);

源碼查看:

簡單的為空返回自己定義的,不為空直接返回!

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

11、orElseGet(Supplier<? extends T> other)

返回值(如果存在),否則調用other并返回該調用的結果。

區別: orElse方法將傳入的參數作為默認值,orElseGet方法可以接受Supplier接口的實現用來生成默認值

如果沒有復雜操作,Idea也會提醒我們不要使用這個,使用orElse即可!

String message = null;
String messageNew = "關注公眾號:小王博客基地";
String orElseGet = Optional.ofNullable(message).orElseGet(() -> "這還是一個空的字符串");
log.info("orElseGet調用:這是空字符串打印的:{}",orElseGet);
String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "這還是一個空的字符串");
log.info("orElseGet調用:這是字符串打印的:{}",orElseGetString);

源碼查看:

和orElse一樣,只不過為空調用lambda執行!

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

12、orElseThrow(Supplier<? extends X> exceptionSupplier)

返回包含的值(如果存在),否則拋出由提供的供應商創建的異常。

String message = null;
String messageNew = "關注公眾號:小王博客基地";
Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));
Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));

我們可以自定義異常,然后來引用!

源碼查看:

為空則走自己寫的異常!

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

13、例子匯總

/**
 * @author wangzhenjun
 * @date 2023/2/27 10:22
 */
@Slf4j
public class OptionalTest {

    public static void main(String[] args) {

        Optional<Object> empty = Optional.empty();
        log.info("empty值:{}",empty);


        Test testNew  = new Test();
        Test test = null;
        Optional<Test> optionalNew = Optional.of(testNew);
        log.info(" optional對象:{}",optionalNew);
//        Optional<Test> optional = Optional.of(test);

        Optional<Test> optionalTest = Optional.ofNullable(test);
        log.info(" optional對象中的ofNullable方法返回值:{}",optionalTest);
        Optional<Test> optionalTestNew = Optional.ofNullable(testNew);
        log.info(" optional對象中的ofNullable方法new返回值:{}",optionalTestNew);

        Test test2 = optionalTestNew.get();
        log.info("原來有值的:經過Optional包裝后get后得到原來的值:{}",test2);
        // Test test1 = optionalTest.get();
        // log.info("原來沒有值的:經過Optional包裝后get后得到原來的值:{}",test1);

        boolean present = optionalTestNew.isPresent();
        log.info("optionalTestNew調用是否為空:{}",present);
        boolean present1 = optionalTest.isPresent();
        log.info("optionalTest調用是否為空:{}",present1);

        optionalTest.ifPresent(new Consumer<Test>() {
            @Override
            public void accept(Test test) {
                log.info("我是調用ifPresent執行后的打印=====");
            }
        });
        optionalTestNew.ifPresent(testInner -> log.info("我是調用ifPresent執行后的打印"));

        testNew.setName("蕭炎");
        testNew.setAge(33);
        Optional<Test> optionalTest1 = optionalTestNew.filter(test1 -> test1.getAge() > 30);
        log.info("過濾后的結果:{}",optionalTest1.get());

        Optional<String> stringOptional = optionalTestNew.map(Test::getName);
        log.info("map后獲得字段值:{}",stringOptional.get());

        Optional<String> optional = optionalTestNew.flatMap(OptionalTest::getFlatMap);
        log.info("flatMap后得到的字段:{}",optional.get());

        String message = null;
        String messageNew = "關注公眾號:小王博客基地";

        String nullString = Optional.ofNullable(message).orElse("這是一個空字符串!");
        log.info("這是空字符串打印的:{}",nullString);
        String string = Optional.ofNullable(messageNew).orElse("=====這是一個空字符串!");
        log.info("這是字符串打印的:{}",string);

        String orElseGet = Optional.ofNullable(message).orElseGet(() -> "這還是一個空的字符串");
        log.info("orElseGet調用:這是空字符串打印的:{}",orElseGet);
        String orElseGetString = Optional.ofNullable(messageNew).orElseGet(() -> "這還是一個空的字符串");
        log.info("orElseGet調用:這是字符串打印的:{}",orElseGetString);

        Optional.ofNullable(messageNew).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));
        Optional.ofNullable(message).orElseThrow(() -> new RuntimeException("為空了,還不看看!"));



    }

    private static Optional<String> getFlatMap(Test test){
        return Optional.ofNullable(test).map(Test::getName);
    }

}

四、總結

這里就不在演示實戰了,基本上組合使用:

Optional.ofNullable(需要判斷的對象).ifPresent(具體操作)。

其實和if相比就是顯得優雅一些,主要是防止某處沒考慮到,忘記if判斷,那么后續可能會導致空指針,如果使用Optional的話,那么這個問題能夠得到避免。

就像多使用設計模式一樣,讓自己的代碼更加健壯優雅,還是要多使用一些的!當然不能過渡使用!!

責任編輯:姜華 來源: 小王博客基地
相關推薦

2023-10-08 11:09:22

Optional空指針

2024-06-19 10:04:15

ifC#代碼

2024-10-15 15:58:11

2024-08-12 08:28:35

2025-02-27 09:39:56

JavaJava 8對象

2021-11-15 06:56:45

系統運行空指針

2024-12-06 10:12:20

2018-07-23 08:19:26

編程語言Python工具

2024-02-01 12:09:17

Optional容器null

2024-02-28 09:03:20

Optional空指針Java

2025-06-16 10:00:00

C++代碼Python

2024-12-02 07:00:00

特性標記軟件開發Action

2016-08-30 10:39:44

云計算

2016-09-01 15:02:38

混合云多云基礎架構

2018-09-18 16:20:08

Asyncjavascript前端

2024-10-14 13:22:15

2025-06-19 09:13:10

空指針編程C++17

2022-07-11 10:51:25

Java 8OptionalNPE

2020-05-14 18:05:55

OptionalJava非空判斷

2022-07-22 09:15:07

OpitonalJava代碼
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久这里有精品 | 羞羞视频免费在线 | 精品一区二区在线观看 | 美女久久 | 日日网| av大片| 免费观看www7722午夜电影 | 国产精品视频不卡 | 日韩综合在线 | 成人欧美一区二区三区在线观看 | 久久久91精品国产一区二区三区 | 亚洲免费高清 | 农村真人裸体丰满少妇毛片 | 91在线视频播放 | 激情一区二区三区 | 中文字幕国产 | 久久久久国产一级毛片高清网站 | 成人免费观看网站 | 呦呦在线视频 | 免费一区在线 | 丁香五月网久久综合 | 玖玖精品| 亚洲人在线播放 | 成人精品鲁一区一区二区 | 欧美久久久久 | aaa大片免费观看 | 国产一区二区日韩 | 亚洲免费在线观看 | 中文字幕在线看 | 天天综合网91 | 成人av电影天堂 | 国产网站在线播放 | 中文在线视频观看 | 麻豆av片 | 狠狠综合久久av一区二区小说 | 天天玩夜夜操 | 亚洲毛片在线观看 | 国产精品视频一区二区三区不卡 | 成人在线观看免费视频 | 天天色图 | 亚洲精品一区二三区不卡 |