溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java判空如何實(shí)現(xiàn)

發(fā)布時(shí)間:2022-05-23 14:53:58 來(lái)源:億速云 閱讀:253 作者:iii 欄目:大數(shù)據(jù)

本文小編為大家詳細(xì)介紹“Java判空如何實(shí)現(xiàn)”,內(nèi)容詳細(xì),步驟清晰,細(xì)節(jié)處理妥當(dāng),希望這篇“Java判空如何實(shí)現(xiàn)”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來(lái)學(xué)習(xí)新知識(shí)吧。

NullObject模式

對(duì)于項(xiàng)目中無(wú)數(shù)次的判空,對(duì)代碼質(zhì)量整潔度產(chǎn)生了十分之惡劣的影響,對(duì)于這種現(xiàn)象,我們稱之為“判空災(zāi)難”。

那么,這種現(xiàn)象如何治理呢,你可能聽(tīng)說(shuō)過(guò)NullObject模式,不過(guò)這不是我們今天的武器,但是還是需要介紹一下NullObject模式。

什么是NullObject模式呢?

In  object-oriented computer programming, a null object is an object with  no referenced value or with defined neutral ("null") behavior. The null  object design pattern describes the uses of such objects and their  behavior (or lack thereof).

以上解析來(lái)自Wikipedia。

NullObject模式首次發(fā)表在“ 程序設(shè)計(jì)模式語(yǔ)言 ”系列叢書(shū)中。一般的,在面向?qū)ο笳Z(yǔ)言中,對(duì)對(duì)象的調(diào)用前需要使用判空檢查,來(lái)判斷這些對(duì)象是否為空,因?yàn)樵诳找蒙蠠o(wú)法調(diào)用所需方法。

示例代碼如下(命名來(lái)自網(wǎng)絡(luò),哈哈到底是有多懶):

Nullable是空對(duì)象的相關(guān)操作接口,用于確定對(duì)象是否為空,因?yàn)樵诳諏?duì)象模式中,對(duì)象為空會(huì)被包裝成一個(gè)Object,成為Null Object,該對(duì)象會(huì)對(duì)原有對(duì)象的所有方法進(jìn)行空實(shí)現(xiàn)…

public interface Nullable {      boolean isNull(); }

這個(gè)接口定義了業(yè)務(wù)對(duì)象的行為。

 
public interface DependencyBase extends Nullable {      void Operation();  }

這是該對(duì)象的真實(shí)類(lèi),實(shí)現(xiàn)了業(yè)務(wù)行為接口DependencyBase與空對(duì)象操作接口Nullable。

public class Dependency implements DependencyBase, Nullable {      @Override     public void Operation() {         System.out.print("Test!");     }      @Override     public boolean isNull() {         return false;     }  }

這是空對(duì)象,對(duì)原有對(duì)象的行為進(jìn)行了空實(shí)現(xiàn)。

public class NullObject implements DependencyBase{      @Override     public void Operation() {         // do nothing     }      @Override     public boolean isNull() {         return true;     }  }

在使用時(shí),可以通過(guò)工廠調(diào)用方式來(lái)進(jìn)行空對(duì)象的調(diào)用,也可以通過(guò)其他如反射的方式對(duì)對(duì)象進(jìn)行調(diào)用(一般多耗時(shí)幾毫秒)在此不進(jìn)行詳細(xì)敘述。

public class Factory {      public static DependencyBase get(Nullable dependencyBase){         if (dependencyBase == null){             return new NullObject();         }         return new Dependency();     }  }

這是一個(gè)使用范例,通過(guò)這種模式,我們不再需要進(jìn)行對(duì)象的判空操作,而是可以直接使用對(duì)象,也不必?fù)?dān)心NPE(NullPointerException)的問(wèn)題。

public class Client {      public void test(DependencyBase dependencyBase){         Factory.get(dependencyBase).Operation();     }  }
.NR Null Object

NR  Null Object是一款適用于Android Studio、IntelliJ  IDEA、PhpStorm、WebStorm、PyCharm、RubyMine、AppCode、CLion、GoLand、DataGrip等IDEA的Intellij插件。其可以根據(jù)現(xiàn)有對(duì)象,便捷快速生成其空對(duì)象模式需要的組成成分,其包含功能如下:

  1. 分析所選類(lèi)可聲明為接口的方法;

  2. 抽象出公有接口;

  3. 創(chuàng)建空對(duì)象,自動(dòng)實(shí)現(xiàn)公有接口;

  4. 對(duì)部分函數(shù)進(jìn)行可為空聲明;

  5. 可追加函數(shù)進(jìn)行再次生成;

  6. 自動(dòng)的函數(shù)命名規(guī)范

讓我們來(lái)看一個(gè)使用范例:

Java判空如何實(shí)現(xiàn)

怎么樣,看起來(lái)是不是非??焖俦憬荩恍枰谠行枰M(jìn)行多次判空的對(duì)象中,郵件彈出菜單,選擇Generate,并選擇NR Null Object即可自動(dòng)生成相應(yīng)的空對(duì)象組件。

那么如何來(lái)獲得這款插件呢?
安裝方式

可以直接通過(guò)IDEA的Preferences中的Plugins倉(cāng)庫(kù)進(jìn)行安裝。

選擇 Preferences → Plugins → Browse repositories

Java判空如何實(shí)現(xiàn)

搜索“NR Null Oject”或者“Null Oject”進(jìn)行模糊查詢,點(diǎn)擊右側(cè)的Install,restart IDEA即可。

Java判空如何實(shí)現(xiàn)

Optional

還有一種方式是使用Java8特性中的Optional來(lái)進(jìn)行優(yōu)雅地判空,Optional來(lái)自官方的介紹如下:

A  container object which may or may not contain a non-null value. If a  value is present, isPresent() will return true and get() will return the  value.

一個(gè)可能包含也可能不包含非null值的容器對(duì)象。如果存在值,isPresent()將返回true,get()將返回該值。

話不多說(shuō),舉個(gè)例子。

有如下代碼,需要獲得Test2中的Info信息,但是參數(shù)為T(mén)est4,我們要一層層的申請(qǐng),每一層都獲得的對(duì)象都可能是空,最后的代碼看起來(lái)就像這樣。  

public String testSimple(Test4 test) {        if (test == null) {            return "";        }        if (test.getTest3() == null) {            return "";        }        if (test.getTest3().getTest2() == null) {            return "";        }        if (test.getTest3().getTest2().getInfo() == null) {            return "";        }        return test.getTest3().getTest2().getInfo();    }

但是使用Optional后,整個(gè)就都不一樣了。    

public String testOptional(Test test) {         return Optional.ofNullable(test).flatMap(Test::getTest3)                 .flatMap(Test3::getTest2)                 .map(Test2::getInfo)                 .orElse("");     }

1、Optional.ofNullable(test),如果test為空,則返回一個(gè)單例空Optional對(duì)象,如果非空則返回一個(gè)Optional包裝對(duì)象,Optional將test包裝;    

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

2、flatMap(Test::getTest3)判斷test是否為空,如果為空,繼續(xù)返回第一步中的單例Optional對(duì)象,否則調(diào)用Test的getTest3方法;    

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

3、flatMap(Test3::getTest2)同上調(diào)用Test3的getTest2方法;

4、map(Test2::getInfo)同flatMap類(lèi)似,但是flatMap要求Test3::getTest2返回值為Optional類(lèi)型,而map不需要,flatMap不會(huì)多層包裝,map返回會(huì)再次包裝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));        }    }

5、orElse("");獲得map中的value,不為空則直接返回value,為空則返回傳入的參數(shù)作為默認(rèn)值。

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

怎么樣,使用Optional后我們的代碼是不是瞬間變得非常整潔,或許看到這段代碼你會(huì)有很多疑問(wèn),針對(duì)復(fù)雜的一長(zhǎng)串判空,Optional有它的優(yōu)勢(shì),但是對(duì)于簡(jiǎn)單的判空使用Optional也會(huì)增加代碼的閱讀成本、編碼量以及團(tuán)隊(duì)新成員的學(xué)習(xí)成本。畢竟Optional在現(xiàn)在還并沒(méi)有像RxJava那樣流行,它還擁有一定的局限性。

如果直接使用Java8中的Optional,需要保證安卓API級(jí)別在24及以上。

Java判空如何實(shí)現(xiàn)

你也可以直接引入Google的Guava。(啥是Guava?來(lái)自官方的提示)

Guava  is a set of core libraries that includes new collection types (such as  multimap and multiset), immutable collections, a graph library,  functional types, an in-memory cache, and APIs/utilities for  concurrency, I/O, hashing, primitives, reflection, string processing,  and much more!

引用方式,就像這樣:    

dependencies {       compile 'com.google.guava:guava:27.0-jre'       // or, for Android:       api 'com.google.guava:guava:27.0-android'     }

不過(guò)IDEA默認(rèn)會(huì)顯示黃色,提示讓你將Guava表達(dá)式遷移到Java Api上。

Java判空如何實(shí)現(xiàn)

當(dāng)然,你也可以通過(guò)在Preferences搜索"Guava"來(lái)Kill掉這個(gè)Yellow的提示。

Java判空如何實(shí)現(xiàn)

使用Optional具有如下優(yōu)點(diǎn):
  1. 將防御式編程代碼完美包裝

  2. 鏈?zhǔn)秸{(diào)用

  3. 有效避免程序代碼中的空指針

但是也同樣具有一些缺點(diǎn):
  1. 流行性不是非常理想,團(tuán)隊(duì)新成員需要學(xué)習(xí)成本

  2. 安卓中需要引入Guava,需要團(tuán)隊(duì)每個(gè)人處理IDEA默認(rèn)提示,或者忍受黃色提示

當(dāng)然,Kotlin以具有優(yōu)秀的空安全性為一大特色,并可以與Java很好的混合使用,like this:    

test1?.test2?.test3?.test4

讀到這里,這篇“Java判空如何實(shí)現(xiàn)”文章已經(jīng)介紹完畢,想要掌握這篇文章的知識(shí)點(diǎn)還需要大家自己動(dòng)手實(shí)踐使用過(guò)才能領(lǐng)會(huì),如果想了解更多相關(guān)內(nèi)容的文章,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI