欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java14發(fā)布了,再也不怕NullPointerException了

 更新時間:2020年03月26日 10:44:42   作者:Hollis  
這篇文章主要介紹了Java14發(fā)布了,再也不怕NullPointerException了,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

2020年3月17日發(fā)布,Java正式發(fā)布了JDK 14 ,目前已經(jīng)可以開放下載。在JDK 14中,共有16個新特性,本文主要來介紹其中的一個特性:JEP 358: Helpful NullPointerExceptions

null何錯之有?

對于Java程序員來說,null是令人頭痛的東西。時常會受到空指針異常(NullPointerException)的騷擾。相信很多程序員都特別害怕出現(xiàn)程序中出現(xiàn)NPE,因為這種異常往往伴隨著代碼的非預期運行。

在編程語言中,空引用(Null Reference)是一個與空指針類似的概念,是一個已宣告但其并未引用到一個有效對象的變量。

在Java 1 中就包含了了Null引用和NPE了,但是其實,Null引用是偉大的計算機科學家Tony Hoare 早在1965年發(fā)明的,最初作為編程語言ALGOL W的一部分。

1965年,英國一位名為Tony Hoare的計算機科學家在設(shè)計ALGOL W語言時提出了null引用的想法。ALGOL W是第一批在堆上分配記錄的類型語言之一。Hoare選擇null引用這種方式,“只是因為這種方法實現(xiàn)起來非常容易”。雖然他的設(shè)計初衷就是要“通過編譯器的自動檢測機制,確保所有使用引用的地方都是絕對安全的”,他還是決定為null引用開個綠燈,因為他認為這是為“不存在的值”建模最容易的方式。

但是在2009年,很多年后,他開始為自己曾經(jīng)做過這樣的決定而后悔不已,把它稱為“一個價值十億美元的錯誤”。實際上,Hoare的這段話低估了過去五十年來數(shù)百萬程序員為修復空引用所耗費的代價。因為在ALGOL W之后出現(xiàn)的大多數(shù)現(xiàn)代程序設(shè)計語言,包括Java,都采用了同樣的設(shè)計方式,其原因是為了與更老的語言保持兼容,或者就像Hoare曾經(jīng)陳述的那樣,“僅僅是因為這樣實現(xiàn)起來更加容易”。

相信很多Java程序員都一樣對null和NPE深惡痛絕,因為他確實會帶來各種各樣的問題(來自《Java 8 實戰(zhàn)》)。如:

  • 它是錯誤之源。 NullPointerException是目前Java程序開發(fā)中最典型的異常。它會使你的代碼膨脹。
  • 它讓你的代碼充斥著深度嵌套的null檢查,代碼的可讀性糟糕透頂。
  • 它自身是毫無意義的。 null自身沒有任何的語義,尤其是是它代表的是在靜態(tài)類型語言中以一種錯誤的方式對缺失變量值的建模。
  • 它破壞了Java的哲學。 Java一直試圖避免讓程序員意識到指針的存在,唯一的例外是:null指針。
  • 它在Java的類型系統(tǒng)上開了個口子。 null并不屬于任何類型,這意味著它可以被賦值給任意引用類型的變量。這會導致問題, 原因是當這個變量被傳遞到系統(tǒng)中的另一個部分后,你將無法獲知這個null變量最初賦值到底是什么類型。

其他語言如何解決NPE問題

我們知道,出了Java語言外,還有很多其他的面向?qū)ο笳Z言,那么在其他的一些語言中,是如何解決NPE的問題的呢?

如在Groovy中使用安全導航操作符(Safe Navigation Operator)可以訪問可能為null的變量:

def carInsuranceName = person?.car?.insurance?.name

Groovy的安全導航操作符能夠避免在訪問這些可能為null引用的變量時發(fā)生NullPointerException,在調(diào)用鏈中的變量遭遇null時將null引用沿著調(diào)用鏈傳遞下去,返回一個null。

其實這個功能曾經(jīng)考慮過增加一個類似的功能,但是后來又被舍棄了。

另外,在Haskell和Scala也有類似的替代品,如Haskell中的Maybe類型、Scala中的Option[T]。

在 Kotlin 中,其類型系統(tǒng)嚴格區(qū)分一個引用可以容納 null 還是不能容納。也就是說,一個變量是否可空必須顯示聲明,對于可空變量,在訪問其成員時必須做空處理,否則無法編譯通過:

var a: String = "abc"
a = null // 編譯錯誤

果允許為空,可以聲明一個可空字符串,寫作 String?:

var b: String? = "abc" //String? 表示該 String 類型變量可為空
b = null // 編譯通過

看到這個?的時候,是不是發(fā)現(xiàn)和Groovy有點像?不過還是有一定區(qū)別的,這里就不展開了。

好了,書歸正傳,我們來看看作為一個TOIBE編程語言排行榜第一名的語言,Java語言對于NPE做出了哪些努力!

Java做了哪些努力

一直以來對于null和NPE的改進還是做出了一些努力的。

首先在Java 8中提供了Optional,其實在Java 8 推出之前,Google的Guava庫中就率先提供過Optional接口來使null快速失敗。

Optional在可能為null的對象上做了一層封裝,Optional對象包含了一些方法來顯式地處理某個值是存在還是缺失,Optional類強制你思考值不存在的情況,這樣就能避免潛在的空指針異常。

但是設(shè)計Optional類的目的并不是完全取代null,它的目的是設(shè)計更易理解的API。通過Optional,可以從方法簽名就知道這個函數(shù)有可能返回一個缺失的值,這樣強制你處理這些缺失值的情況。

關(guān)于Optional的用法,不是本文的重點,就不在這里詳細介紹了,筆者在日常開發(fā)中經(jīng)常結(jié)合Stream一起使用Optional,還是比較好用的。

另外一個值得一提的就是最近(2020年03月17日)發(fā)布的JDK 14中對于NPE有了一個增強。那就是JEP 358: Helpful NullPointerExceptions

更有幫助的NPE

JDK 14中對于NEP有了一個增強,既然NPE暫時無法避免,那么就讓他對開發(fā)者更有幫助一些。

每個Java開發(fā)人員都遇到過NullPointerExceptions (NPEs)。由于NPEs可以發(fā)生在程序的幾乎任何地方,試圖捕獲并從它們中恢復通常是不切實際的。因此,開發(fā)人員通常依賴于JVM來確定NPE實際發(fā)生時的來源。例如,假設(shè)在這段代碼中出現(xiàn)了一個NPE:

a.i = 99;

JVM將打印出導致NPE的方法、文件名和行號:

Exception in thread "main" java.lang.NullPointerException
at Prog.main(Prog.java:5)

通過以上堆棧信息,開發(fā)人員可以定位到a.i= 99這一行,并推斷出a一定是null。

但是,對于更復雜的代碼,如果不使用調(diào)試器,就不可能確定哪個變量是null。假設(shè)在這段代碼中出現(xiàn)了一個NPE:

a.b.c.i = 99;

我們根本無法確定到底是a還是b或者是c在運行時是個null值。

但是,在JDK14以后,這種窘境就有解了。

在JDK14中,當運行期,試圖對一個bull對象進行應用時,JVM依然會拋出一個NullPointerException (NPE),除此之外,還會通過通過分析程序的字節(jié)碼指令,JVM將精確地確定哪個變量是null,并且在堆棧信息中明確的提示出來。

在JDK 14中,如果上文中的a.i = 99發(fā)生NPE,將會打印如下堆棧:

Exception in thread "main" java.lang.NullPointerException:
Cannot assign field "i" because "a" is null
at Prog.main(Prog.java:5)

如果是a.b.c.i = 99;中的b為null導致了空指針,則會打印以下堆棧信息:

Exception in thread "main" java.lang.NullPointerException:
Cannot read field "c" because "a.b" is null
at Prog.main(Prog.java:5)

可見,堆棧中明確指出了到底是哪個對象為null而導致了NPE,這樣,一旦應用中發(fā)生NPE,開發(fā)者可以通過堆棧信息第一時間定位到到底是代碼中的那個對象為null導致的。

這算是JDK的一個小小的改進,但是這個改進對于開發(fā)者來說確實是非常友好的。真的希望這些小而美的改動可以在JDK中越來越多。

參考資料:

https://openjdk.java.net/jeps/358

《Java 8 In Action》

到此這篇關(guān)于Java14發(fā)布了,再也不怕NullPointerException了的文章就介紹到這了,更多相關(guān)Java14 NullPointerException內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java設(shè)計模式之備忘錄模式

    Java設(shè)計模式之備忘錄模式

    這篇文章介紹了Java設(shè)計模式之備忘錄模式,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-10-10
  • Spring通過工具類實現(xiàn)獲取容器中的Bean

    Spring通過工具類實現(xiàn)獲取容器中的Bean

    在實際開發(fā)中,我們往往要用到Spring容器為我們提供的諸多資源,例如想要獲取到容器中的配置、獲取到容器中的Bean等等。本文為大家詳細講講工具類如何獲取到Spring容器中的Bean,需要的可以參考一下
    2022-06-06
  • 使用@NonNull注解遇到的小問題及解決

    使用@NonNull注解遇到的小問題及解決

    這篇文章主要介紹了使用@NonNull注解遇到的小問題及解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • Java實現(xiàn)調(diào)用ElasticSearch?API的示例詳解

    Java實現(xiàn)調(diào)用ElasticSearch?API的示例詳解

    這篇文章主要為大家詳細介紹了Java調(diào)用ElasticSearch?API的效果資料,文中的示例代碼講解詳細,具有一定的參考價值,感興趣的可以了解一下
    2023-03-03
  • Java連接MySQL數(shù)據(jù)庫并實現(xiàn)數(shù)據(jù)交互的示例

    Java連接MySQL數(shù)據(jù)庫并實現(xiàn)數(shù)據(jù)交互的示例

    數(shù)據(jù)庫是非常重要的一種存儲格式,可以大大提高存儲效率,本文主要介紹了Java連接MySQL數(shù)據(jù)庫并實現(xiàn)數(shù)據(jù)交互的示例,具有一定的參考價值,感興趣的可以了解一下
    2024-03-03
  • spring boot 全局異常處理方法匯總

    spring boot 全局異常處理方法匯總

    這篇文章主要介紹了spring boot 全局異常處理方法匯總,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-10-10
  • java實現(xiàn)163郵箱發(fā)送郵件到qq郵箱成功案例

    java實現(xiàn)163郵箱發(fā)送郵件到qq郵箱成功案例

    這篇文章主要為大家分享了java實現(xiàn)163郵箱發(fā)送郵件到qq郵箱成功案例,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2016-05-05
  • 關(guān)于@Autowired注解和靜態(tài)方法及new的關(guān)系

    關(guān)于@Autowired注解和靜態(tài)方法及new的關(guān)系

    這篇文章主要介紹了關(guān)于@Autowired注解和靜態(tài)方法及new的關(guān)系,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • JAVA MyBatis入門學習過程記錄

    JAVA MyBatis入門學習過程記錄

    MyBatis是一個支持普通SQL查詢,存儲過程和高級映射的優(yōu)秀持久層框架。這篇文章主要介紹了mybatis框架入門學習教程,需要的朋友可以參考下,希望能幫助到你
    2021-06-06
  • Java中怎樣處理空指針異常

    Java中怎樣處理空指針異常

    程序中的變量是?null,就意味著它沒有引用指向或者說沒有指針。這時,我們對這個變量進行任何操作,都必然會引發(fā)空指針異常,本文主要介紹了Java中怎樣處理空指針異常,感興趣的可以了解一下
    2022-05-05

最新評論