Null is your friend, not a mistake

来源:https://www.cnblogs.com/bingxinshuo/archive/2019/09/18/11546151.html
-Advertisement-
Play Games

原文作者: "Roman Elizarov" 原文地址: "Null is your friend, not a mistake" 譯者:秉心說 "Kotlin Island from Wikimedia by Pavlikhin, CC BY SA 4.0" 我使用 Java 語言編程已經很久很久 ...


原文作者: Roman Elizarov

原文地址: Null is your friend, not a mistake

譯者:秉心說

Kotlin Island from Wikimedia by Pavlikhin, CC BY-SA 4.0

我使用 Java 語言編程已經很久很久了,掌握了通過 Java 編寫和維護大型軟體(百萬行代碼)應該註意些什麼,並親眼目睹了全行業都在竭力避免空指針異常 NullPointerException(NPE),它困擾著大大小小的 Java 類庫。在 2009 年其發明者 Tony Hoare 承認空引用是他造成的一個 “十億美元的錯誤”之前,人人已經意識到它的危險性。

在 1996 年 Java 1.0 發佈時,這個問題還不是如此明顯。讓我們看一個典型的 Java API 的例子:File.list() 方法。它被用來列舉文件夾中的內容,如下所示:

for(String name : new File("directory").list()) {
    System.out.println(name);
}

僅當文件夾存在時上面的代碼才會正常運行,否則將拋出 NPE,因為 list() 返回了 null。但是誰會寫這樣的代碼呢?不僅僅 list() 方法的文檔中清楚的說明瞭文件夾不存在時將返回 null,而且現代的 IDE 也會對特定的代碼給你提出警告。

但是,當使用 Java 編程時,開發者經常犯這類錯誤。到目前為止,有大量研究表明它們是如何發生的。結果表明,大多數情況下,我們的 API 函數不應該返回 null,其他開發者也並不希望返回 null。在一些特殊情況下,比如預設值,Java 中的慣例是返回一些 “空對象”(空集合,未填充的對象等等),或者拋出異常,比返回 null 更糟糕。這就是為什麼要設計 Files.newDirectoryStream, 高級版本的 File.list,任何情況都不會出現 null。

所以當 null 在一些特殊情況下作為函數返回值時,如性能優化,未初始化的引用欄位等等,它常常會讓你措手不及,沒有做好準備去處理它。不僅僅是必須處理空值的情況很少,而且在 Java 中用來處理空值的代碼是很啰嗦的:

String[] list = new File("directory").list();
if (list != null) {
    for (String name : list) {
        System.out.println(name);
    }
}

毫無疑問,除非真的需要(你的客戶在生產環境發現了 NPE),不然你真的不想寫這樣的代碼。

對 null 的恐懼導致了一些極端情況。有一些 Java 編碼風格完全禁止 null,將可惡的工作交給開發者。不知道你有沒有見過這樣的 Java 類庫,所有的域對象都要實現一個特殊的 Null 介面,並且提供手動編碼生成的 “空對象” 實例。如果沒有見過說明你還是幸運的。但是我敢打賭你已經看到了只為了避免空值而污染 Java 代碼的 Optional <T> 包裝器(譯者註:Java 8 新特性)。

有些集合框架的 API 出於謹慎禁止 null 元素,並且一些 Java 核心團隊成員認為 Java 集合框架對 null 的支持是一個錯誤。這讓人非常難過。

事實上,null 這個概念不是一個錯誤,但是 Java 的類型系統認為 null 是任何類型的成員。 讓我們看看,在 Java 中 “abc” 是一個合法的 Stringnull 也是一個合法的 String。你可以在前者上使用 string 的所有方法,例如 substring。但是在後者上使用則會發生運行時錯誤。它是類型安全的嗎?並不完全是。一個類型的特定值在進行某些操作時發生運行時異常(例如除 0)是正常的,但是當對一個值進行該類型的所有操作都發生了異常,這首先表明的是這個值並不屬於這個類型。所有的那些 NPE 都表明瞭 Java 的類型系統存在明顯的缺陷。

更多的類型安全的編程語言,例如 Kotlin,通過合理的將 null 的概念合併到類型系統中來修複這個缺陷。添加檢查和警告也有一定作用,但這並不夠。顯然,一個健全的類型系統必須允許 String 類型的所有變數都支持它的操作。所以在 Kotlin 中,將 null 賦給 String 類型的變數就不僅僅只是一個警告了,而是類型錯誤,就像把數值 42 賦給 String 類型變數一樣。

類型系統中合理的 null 支持是 API 設計的一個轉折點,沒有任何理由再去害怕 null 了。一些函數返回可空類型 String?,另一些函數返回不可空類型 String,就和一些函數返回 String,另一些返回 Int 一樣。它們都是不同的類型,有著不同但是安全的操作集。

用類型安全的 null 來表示 “缺失的值” 是更好,更高效,更簡潔的。看一下 Kotlin 標準庫中的 String.toIntOrNull() 函數,用於將 string 轉為數字,不能轉換的話返回 null。使用起來很愉快,編寫一個命令行程式,接受 integer 參數並處理參數的缺失就很簡單:

fun main(args: Array<String>) {
    val id = args.getOrNull(0)?.toIntOrNull() 
        ?: error("id expected")
    // ...
}

在 API 設計中使用 null 吧,它是你和 Kotlin 的好朋友。沒有理由去害怕它,也沒有理由使用 null object 模式或者包裝器來處理它,更不用說異常了。在你的 API 中合理使用 null 會給你帶來更可讀,更安全的代碼,並且遠離樣板代碼。

深入閱讀

如果你喜歡這個主題,並且想瞭解更多關於語言設計的細節,那麼可以考慮閱讀這篇文章—— Dealing with absence of value

文章首發微信公眾號: 秉心說 , 專註 Java 、 Android 原創知識分享,LeetCode 題解。

更多最新原創文章,掃碼關註我吧!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、 Redis 介紹 Remote Dictionary Server(Redis)是一個開源的使用 ANSI C 語言編寫、支持網路、可基於記憶體亦可持久化的日誌型、Key-Value 資料庫,並提供多種語言的 API。它通常被稱為數據結構伺服器,因為值(value)可以是 字元串(String) ...
  • [toc] 發表日期:2019年9月19日 上節回顧 在學習新的內容之前,先回顧一下上節的內容,上節主要講述了以下的內容: 1. ElasticSearch是什麼?什麼是搜索引擎?為什麼選擇ElasticSearch? 2. 搜索是怎麼做到的:分詞、倒排索引? 3. 環境的搭建 4. 如何通過kib ...
  • Linux下如何設置每天自動備份資料庫 本文以Centos7.6系統與Oracle11g為例: 一.先找到資料庫的環境變數 如果是在root賬戶下,須先登錄到資料庫所在賬戶 export PATHexport ORACLE_BASE=/home/nnc_db/appexport ORACLE_HOM ...
  • 首先在命令提示符下進入mysqldump.exe所在目錄(如果mysqldump.exe所在目錄已添加到系統path環境變數,可以省略此步驟) 備份mysqldump.exe --opt --add-drop-database --add-drop-table -hlocalhost -uroot ...
  • SELECT INTO 語句可用於創建表的備份復件。 SELECT INTO 語句 SELECT INTO 語句從一個表中選取數據,然後把數據插入另一個表中。 SELECT INTO 語句常用於創建表的備份復件或者用於對記錄進行存檔。 SQL SELECT INTO 語法 您可以把所有的列插入新表: ...
  • 本文鏈接: "Android mmap 文件映射到記憶體介紹" Android開發中,我們可能需要記錄一些文件。例如記錄log文件。如果使用流來寫文件,頻繁操作文件io可能會引起性能問題。 為了降低寫文件的頻率,我們可能會採用緩存一定數量的log,再一次性把它們寫到文件中。如果app異常退出,我們有可 ...
  • 關於小程式的載入快慢這可是一大學問,自古以來性能都是重點,所以下麵我淺談一下自己遇到的問題和解決方法吧 首先,先從網路請求network說起: 這裡基本不關前端的事情,但是這也是優化小程式的一大重點,後端響應我們請求數據的速度影響了整個頁面的速度,所以,把它拿到第一位 請求超過300ms就已經算是慢 ...
  • 最近項目要求,ui有很多有關於陰影的設計要求,網上找了些實現方式,但都不是很理想。現在閑下來了,就尋思著自己寫個陰影佈局耍耍,以備後用。先說道說道我找到的幾種陰影實現方式: 系統陰影 Andorid 系統自api 21之後就多了一個熟悉 android:elevation ,這是android最新引 ...
一周排行
    -Advertisement-
    Play Games
  • 1. 說明 /* Performs operations on System.String instances that contain file or directory path information. These operations are performed in a cross-pla ...
  • 視頻地址:【WebApi+Vue3從0到1搭建《許可權管理系統》系列視頻:搭建JWT系統鑒權-嗶哩嗶哩】 https://b23.tv/R6cOcDO qq群:801913255 一、在appsettings.json中設置鑒權屬性 /*jwt鑒權*/ "JwtSetting": { "Issuer" ...
  • 引言 集成測試可在包含應用支持基礎結構(如資料庫、文件系統和網路)的級別上確保應用組件功能正常。 ASP.NET Core 通過將單元測試框架與測試 Web 主機和記憶體中測試伺服器結合使用來支持集成測試。 簡介 集成測試與單元測試相比,能夠在更廣泛的級別上評估應用的組件,確認多個組件一起工作以生成預 ...
  • 在.NET Emit編程中,我們探討了運算操作指令的重要性和應用。這些指令包括各種數學運算、位操作和比較操作,能夠在動態生成的代碼中實現對數據的處理和操作。通過這些指令,開發人員可以靈活地進行算術運算、邏輯運算和比較操作,從而實現各種複雜的演算法和邏輯......本篇之後,將進入第七部分:實戰項目 ...
  • 前言 多表頭表格是一個常見的業務需求,然而WPF中卻沒有預設實現這個功能,得益於WPF強大的控制項模板設計,我們可以通過修改控制項模板的方式自己實現它。 一、需求分析 下圖為一個典型的統計表格,統計1-12月的數據。 此時我們有一個需求,需要將月份按季度劃分,以便能夠直觀地看到季度統計數據,以下為該需求 ...
  • 如何將 ASP.NET Core MVC 項目的視圖分離到另一個項目 在當下這個年代 SPA 已是主流,人們早已忘記了 MVC 以及 Razor 的故事。但是在某些場景下 SSR 還是有意想不到效果。比如某些靜態頁面,比如追求首屏載入速度的時候。最近在項目中回歸傳統效果還是不錯。 有的時候我們希望將 ...
  • System.AggregateException: 發生一個或多個錯誤。 > Microsoft.WebTools.Shared.Exceptions.WebToolsException: 生成失敗。檢查輸出視窗瞭解更多詳細信息。 內部異常堆棧跟蹤的結尾 > (內部異常 #0) Microsoft ...
  • 引言 在上一章節我們實戰了在Asp.Net Core中的項目實戰,這一章節講解一下如何測試Asp.Net Core的中間件。 TestServer 還記得我們在集成測試中提供的TestServer嗎? TestServer 是由 Microsoft.AspNetCore.TestHost 包提供的。 ...
  • 在發現結果為真的WHEN子句時,CASE表達式的真假值判斷會終止,剩餘的WHEN子句會被忽略: CASE WHEN col_1 IN ('a', 'b') THEN '第一' WHEN col_1 IN ('a') THEN '第二' ELSE '其他' END 註意: 統一各分支返回的數據類型. ...
  • 在C#編程世界中,語法的精妙之處往往體現在那些看似微小卻極具影響力的符號與結構之中。其中,“_ =” 這一組合突然出現還真不知道什麼意思。本文將深入剖析“_ =” 的含義、工作原理及其在實際編程中的廣泛應用,揭示其作為C#語法奇兵的重要角色。 一、下劃線 _:神秘的棄元符號 下劃線 _ 在C#中並非 ...