如何判断map为空_在Java中如何优雅地判空
鏈接:http://blog.imuxuan.com/archives/86
判空災(zāi)難
作為搬磚黨的一族們,我們對(duì)判空一定再熟悉不過(guò)了,不要跟我說(shuō)你很少進(jìn)行判空,除非你喜歡 NullPointerException。
不過(guò) NullPointerException對(duì)于很多猿們來(lái)說(shuō),也是 Exception家族中最親近的一員了。
為了避免 NullPointerException來(lái)找我們,我們經(jīng)常會(huì)進(jìn)行如下操作:
if (data != null) {do sth. }如果一個(gè)類中多次使用某個(gè)對(duì)象,那你可能要一頓操作,so:如果一個(gè)類中多次使用某個(gè)對(duì)象,那你可能要一頓操作,so:
世界第九大奇跡”就這樣誕生了。Maybe你會(huì)想,項(xiàng)目中肯定不止你一個(gè)人會(huì)這樣一頓操作,然后按下 Command+Shift+F,真相就在眼前:
What,我們有接近一萬(wàn)行的代碼都是在判空?
好了,接下來(lái),要進(jìn)入正題了。
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).NullObject模式首次發(fā)表在“ 程序設(shè)計(jì)模式語(yǔ)言 ”系列叢書(shū)中。一般的,在面向?qū)ο笳Z(yǔ)言中,對(duì)對(duì)象的調(diào)用前需要使用判空檢查,來(lái)判斷這些對(duì)象是否為空,因?yàn)樵诳找蒙蠠o(wú)法調(diào)用所需方法。
空對(duì)象模式的一種典型實(shí)現(xiàn)方式如下圖所示(圖片來(lái)自網(wǎng)絡(luò)):
示例代碼如下(命名來(lái)自網(wǎng)絡(luò),哈哈到底是有多懶):
Nullable是空對(duì)象的相關(guān)操作接口,用于確定對(duì)象是否為空,因?yàn)樵诳諏?duì)象模式中,對(duì)象為空會(huì)被包裝成一個(gè) Object,成為 NullObject,該對(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í)類,實(shí)現(xiàn)了業(yè)務(wù)行為接口 DependencyBase與空對(duì)象操作接口 Nullable。
public class Dependency implements DependencyBase, Nullable {@Overridepublic void Operation() {System.out.print("Test!");}@Overridepublic boolean isNull() {return false;}}這是空對(duì)象,對(duì)原有對(duì)象的行為進(jìn)行了空實(shí)現(xiàn)。
public class NullObject implements DependencyBase{@Overridepublic void Operation() {// do nothing}@Overridepublic 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();}}關(guān)于空對(duì)象模式,更具體的內(nèi)容大家也可以多找一找資料,上述只是對(duì) NullObject的簡(jiǎn)單介紹,但是,今天我要推薦的是一款協(xié)助判空的插件 NRNullObject,讓我們來(lái)優(yōu)雅地進(jìn)行判空,不再進(jìn)行一頓操作來(lái)定義繁瑣的空對(duì)象接口與空獨(dú)享實(shí)現(xiàn)類。
.NR Null Object
NRNullObject是一款適用于Android Studio、IntelliJ IDEA、PhpStorm、WebStorm、PyCharm、RubyMine、AppCode、CLion、GoLand、DataGrip等 IDEA的 Intellij插件。其可以根據(jù)現(xiàn)有對(duì)象,便捷快速生成其空對(duì)象模式需要的組成成分,其包含功能如下:
- 分析所選類可聲明為接口的方法;
- 抽象出公有接口;
- 創(chuàng)建空對(duì)象,自動(dòng)實(shí)現(xiàn)公有接口;
- 對(duì)部分函數(shù)進(jìn)行可為空聲明;
- 可追加函數(shù)進(jìn)行再次生成;
- 自動(dòng)的函數(shù)命名規(guī)范
讓我們來(lái)看一個(gè)使用范例:
怎么樣,看起來(lái)是不是非常快速便捷,只需要在原有需要進(jìn)行多次判空的對(duì)象中,郵件彈出菜單,選擇 Generate,并選擇 NRNullObject即可自動(dòng)生成相應(yīng)的空對(duì)象組件。
那么如何來(lái)獲得這款插件呢?安裝方式
可以直接通過(guò) IDEA的 Preferences中的 Plugins倉(cāng)庫(kù)進(jìn)行安裝。
選擇 Preferences→Plugins→Browserepositories
搜索 “NR Null Oject” 或者 “Null Oject”進(jìn)行模糊查詢,點(diǎn)擊右側(cè)的 Install,restart IDEA 即可
Optional
還有一種方式是使用 Java8特性中的 Optional來(lái)進(jìn)行優(yōu)雅地判空。一個(gè)可能包含也可能不包含非null值的容器對(duì)象。如果存在值, isPresent()將返回 true, get()將返回該值。
話不多說(shuō),舉個(gè)例子。有如下代碼,需要獲得 Test2中的 Info信息,但是參數(shù)為 Test4,我們要一層層的申請(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("");}- Optional.ofNullable(test),如果 test為空,則返回一個(gè)單例空 Optional對(duì)象,如果非空則返回一個(gè) Optional包裝對(duì)象, Optional將 test包裝;
- flatMap(Test::getTest3)判斷 test是否為空,如果為空,繼續(xù)返回第一步中的單例 Optional對(duì)象,否則調(diào)用 Test的 getTest3方法;
- flatMap(Test3::getTest2)同上調(diào)用 Test3的 getTest2方法;
- map(Test2::getInfo)同 flatMap類似,但是 flatMap要求 Test3::getTest2返回值為 Optional類型,而 map不需要, flatMap不會(huì)多層包裝, map返回會(huì)再次包裝 Optional;
- orElse("");獲得 map中的 value,不為空則直接返回 value,為空則返回傳入的參數(shù)作為默認(rèn)值。
怎么樣,使用 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及以上。
你也可以直接引入 Google的 Guava。引用方式,就像這樣:
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上。
當(dāng)然,你也可以通過(guò)在 Preferences搜索 "Guava"來(lái) Kill掉這個(gè) Yellow的提示。
關(guān)于 Optional使用還有很多技巧,感興趣可以查閱 Guava和 Java8相關(guān)書(shū)籍和文檔。
使用 Optional具有如下優(yōu)點(diǎn):
- 將防御式編程代碼完美包裝
- 鏈?zhǔn)秸{(diào)用
- 有效避免程序代碼中的空指針
但是也同樣具有一些缺點(diǎn):
- 流行性不是非常理想,團(tuán)隊(duì)新成員需要學(xué)習(xí)成本
- 安卓中需要引入 Guava,需要團(tuán)隊(duì)每個(gè)人處理 IDEA默認(rèn)提示,或者忍受黃色提示
- 有時(shí)候代碼閱讀看起來(lái)可能會(huì)如下圖所示: Kotlin
Kotlin
當(dāng)然,Kotlin以具有優(yōu)秀的空安全性為一大特色,并可以與 Java很好的混合使用,like this:
test1?.test2?.test3?.test4牛!
總結(jié)
以上是生活随笔為你收集整理的如何判断map为空_在Java中如何优雅地判空的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: sql int 转string_SQL智
- 下一篇: tl r402路由器设置_家里新安装宽带