【Java8新特性】05 使用Optional取代null

Java8 由Oracle在2014年發(fā)布奄抽,是繼Java5之后最具革命性的版本爆阶。

Java8吸收其他語言的精髓帶來了函數(shù)式編程窍育,lambda表達(dá)式遂铡,Stream流等一系列新特性,學(xué)會(huì)了這些新特性爬泥,可以讓你實(shí)現(xiàn)高效編碼優(yōu)雅編碼柬讨。

1. 不受待見的空指針異常

有個(gè)小故事:null引用最早是由英國科學(xué)家Tony Hoare提出的,多年后Hoare為自己的這個(gè)想法感到后悔莫及袍啡,并認(rèn)為這是"價(jià)值百萬的重大失誤"踩官。可見空指針是多么不受待見葬馋。

NullPointerException是Java開發(fā)中最常遇見的異常卖鲤,遇到這種異常我們通常的解決方法是在調(diào)用的地方加一個(gè)if判空肾扰。

if判空越多會(huì)造成過多的代碼分支畴嘶,后續(xù)代碼維護(hù)也就越來越復(fù)雜。

2. 糟糕的代碼

比如看下面這個(gè)例子集晚,使用過多的if判空窗悯。

Person對(duì)象里定義了House對(duì)象,House對(duì)象里定義了Address對(duì)象:

publicclassPerson{privateString name;privateintage;privateHouse house;publicHousegetHouse(){returnhouse;}}classHouse{privatelongprice;privateAddress address;publicAddressgetAddress(){returnaddress;}}classAddress{privateString country;privateString city;publicStringgetCity(){returncity;}}

現(xiàn)在獲取這個(gè)人買房的城市偷拔,那么通常會(huì)這樣寫:

publicStringgetCity(){String city=newPerson().getHouse().getAddress().getCity();returncity;}

但是這樣寫容易出現(xiàn)空指針的問題蒋院,比如這個(gè)人沒有房亏钩,House對(duì)象為null。接著你會(huì)改造這段代碼欺旧,加上很多判斷條件:

publicStringgetCity2(Person person){if(person!=null){House house=person.getHouse();if(house!=null){Address address=house.getAddress();if(address!=null){String city=address.getCity();returncity;}}}return"unknown";}

為了避免空指針異常姑丑,每一層都加上判斷,但是這樣會(huì)造成代碼嵌套太深辞友,不易維護(hù)栅哀。

你可能想到如何改造上面的代碼,比如加上提前判空退出:

publicStringgetCity3(Person person){String city="unknown";if(person==null){returncity;}House house=person.getHouse();if(house==null){returncity;}Address address=house.getAddress();if(address==null){returncity;}returnaddress.getCity();}

但是這樣簡單的代碼已經(jīng)加入了三個(gè)退出條件称龙,非常不利于后面代碼維護(hù)留拾。那怎樣才能將代碼寫的優(yōu)雅一點(diǎn)呢,下面引入今天的主角"Optional"鲫尊。

3. 解決空指針的"銀彈"

從Java8開始引入了一個(gè)新類 java.util.Optional痴柔,這是一個(gè)對(duì)象的容器,意味著可能包含或者沒有包含一個(gè)非空的值疫向。下面重點(diǎn)看一下Optional的常用方法:

publicfinalclassOptional<T>{// 通過指定非空值創(chuàng)建Optional對(duì)象// 如果指定的值為null咳蔚,會(huì)拋空指針異常publicstatic<T>Optional<T>of(T value){returnnewOptional<>(value);}// 通過指定可能為空的值創(chuàng)建Optional對(duì)象publicstatic<T>Optional<T>ofNullable(T value){returnvalue==null?empty():of(value);}// 返回值,不存在拋異常publicTget(){if(value==null){thrownewNoSuchElementException("No value present");}returnvalue;}// 如果值存在鸿捧,根據(jù)consumer實(shí)現(xiàn)類消費(fèi)該值publicvoidifPresent(Consumer<?superT>consumer){if(value!=null)consumer.accept(value);}// 如果值存在則返回屹篓,如果值為空則返回指定的默認(rèn)值publicTorElse(T other){returnvalue!=null?value:other;}// map flatmap等方法與Stream使用方法類似,這里不再贅述匙奴,讀者可以參考之前的Stream系列堆巧。}

以上就是Optional類常用的方法,使用起來非常簡單泼菌。

4. Optional使用入門

(1)創(chuàng)建Optional實(shí)例

創(chuàng)建空的Optional對(duì)象谍肤。可以通過靜態(tài)工廠方法Optional.Empty() 創(chuàng)建一個(gè)空的對(duì)象哗伯,例如:

Optional<Person>optionalPerson=Optional.Empty();

指定非空值創(chuàng)建Optional對(duì)象荒揣。

Person person=newPerson();Optional<Person>optionalPerson=Optional.of(person);

指定可能為空的值創(chuàng)建Optional對(duì)象。

Person person=null;// 可能為空Optional<Person>optionalPerson=Optional.of(person);

(2)常用方法

ifPresent

如果值存在焊刹,則調(diào)用consumer實(shí)例消費(fèi)該值系任,否則什么都不執(zhí)行。舉個(gè)栗子:

String str="hello java8";// output: hello java8Optional.ofNullable(str).ifPresent(System.out::println);String str2=null;// output: nothingOptional.ofNullable(str2).ifPresent(System.out::println);

filter, map, flatMap

在三個(gè)方法在前面講Stream的時(shí)候已經(jīng)詳細(xì)講解過虐块,讀者可以翻看之前寫的文章俩滥,這里不再贅述。

orElse

如果value為空贺奠,則返回默認(rèn)值霜旧,舉個(gè)栗子:

publicvoidtest(String city){String defaultCity=Optional.ofNullable(city).orElse("unknown");}

orElseGet

如果value為空,則調(diào)用Supplier實(shí)例返回一個(gè)默認(rèn)值儡率。舉個(gè)例子:

publicvoidtest2(String city){// 如果city為空挂据,則調(diào)用generateDefaultCity方法String defaultCity=Optional.of(city).orElseGet(this::generateDefaultCity);}privateStringgenerateDefaultCity(){return"beijing";}

orElseThrow

如果value為空以清,則拋出自定義異常。舉個(gè)栗子:

publicvoidtest3(String city){// 如果city為空崎逃,則拋出空指針異常掷倔。String defaultCity=Optional.of(city).orElseThrow(NullPointerException::new);}

5. 使用Optional重構(gòu)代碼

再看一遍重構(gòu)之前的代碼,使用了三個(gè)if使代碼嵌套層次變得很深个绍。

// before refactorpublicStringgetCity2(Person person){if(person!=null){House house=person.getHouse();if(house!=null){Address address=house.getAddress();if(address!=null){String city=address.getCity();returncity;}}}return"unknown";}

使用Optional重構(gòu)

publicStringgetCityUsingOptional(Person person){String city=Optional.ofNullable(person).map(Person::getHouse).map(House::getAddress).map(Address::getCity).orElse("Unknown city");returncity;}

只使用了一行代碼就獲取到city值今魔,不用再去不斷的判斷是否為空,這樣寫代碼是不是很優(yōu)雅呀障贸。趕緊用Optional重構(gòu)你的項(xiàng)目吧~

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末错森,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子篮洁,更是在濱河造成了極大的恐慌涩维,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件袁波,死亡現(xiàn)場離奇詭異瓦阐,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)篷牌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門睡蟋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人枷颊,你說我怎么就攤上這事戳杀。” “怎么了夭苗?”我有些...
    開封第一講書人閱讀 164,057評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵信卡,是天一觀的道長。 經(jīng)常有香客問我题造,道長傍菇,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,509評(píng)論 1 293
  • 正文 為了忘掉前任界赔,我火速辦了婚禮丢习,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘淮悼。我一直安慰自己咐低,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,562評(píng)論 6 392
  • 文/花漫 我一把揭開白布敛惊。 她就那樣靜靜地躺著渊鞋,像睡著了一般绰更。 火紅的嫁衣襯著肌膚如雪瞧挤。 梳的紋絲不亂的頭發(fā)上锡宋,一...
    開封第一講書人閱讀 51,443評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音特恬,去河邊找鬼执俩。 笑死,一個(gè)胖子當(dāng)著我的面吹牛癌刽,可吹牛的內(nèi)容都是我干的役首。 我是一名探鬼主播,決...
    沈念sama閱讀 40,251評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼显拜,長吁一口氣:“原來是場噩夢啊……” “哼衡奥!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起远荠,我...
    開封第一講書人閱讀 39,129評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤矮固,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后譬淳,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體档址,經(jīng)...
    沈念sama閱讀 45,561評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,779評(píng)論 3 335
  • 正文 我和宋清朗相戀三年邻梆,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了守伸。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,902評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡浦妄,死狀恐怖尼摹,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情剂娄,我是刑警寧澤窘问,帶...
    沈念sama閱讀 35,621評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站宜咒,受9級(jí)特大地震影響惠赫,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜故黑,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,220評(píng)論 3 328
  • 文/蒙蒙 一儿咱、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧场晶,春花似錦混埠、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春吏颖,著一層夾襖步出監(jiān)牢的瞬間搔体,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評(píng)論 1 269
  • 我被黑心中介騙來泰國打工半醉, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留疚俱,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,025評(píng)論 2 370
  • 正文 我出身青樓缩多,卻偏偏與公主長得像呆奕,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子衬吆,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,843評(píng)論 2 354