Null is your firend, not a mistake

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

原文作者: "Roman Elizarov" 原文地址: "Null is your firend, 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
更多相關文章
  • 目錄結構 mysql 目錄下的 data 為數據目錄,mysql 的數據表、二進位日誌文件就在這裡。.env 文件包含了一些變數,這些變數可以在 docker compose.yml 文件中通過 ${variable_name} 來引用。 當然也可以把 mysql 的目錄放到其它地方,這裡圖個方便, ...
  • ACID特性: 原子性(Atomicity)、一致性(Consistency)、隔離性(Isolation)、持久性(Durability) 原子性:一個事務必須被視為一個不可分割的最小工作單元,整個事務中的所有操作要麼全部提交成功,要麼全部失敗回滾,事務要麼成功(可見),要麼失敗(不可見),不存在 ...
  • 本文基於java構建Flink1.9版本入門程式,需要Maven 3.0.4 和 Java 8 以上版本。需要安裝Netcat進行簡單調試。 這裡簡述安裝過程,並使用IDEA進行開發一個簡單流處理程式,本地調試或者提交到Flink上運行,Maven與JDK安裝這裡不做說明。 一、Flink簡介 Fl ...
  • 第一步:VMware workstation網路設置第二步:虛擬機網路設置一、設置VMware Network AdapterVMnet8的IP1、打開“網路和Internet”設置選擇“更改適配器選項”2、右鍵VMnet8點擊“屬性”3、找到Internet協議版本4(TCP/IPv4),再點擊“ ...
  • 轉自: "https://www.fengjunzi.com/blog 25573.html" 問題 有時候sqlserver無法啟動了,原因是mssqlserver服務沒有啟動,當你手動啟動時,又出現服務無法響應的可惡錯誤提示。。。 筆者“有幸”遇到了,我的原因是第5個,禁用TCP/IP協議,現在 ...
  • 最近有個業務需要用到sql server 的觸發器,網上找了半天,查出來大部分都是oracle 的,決心自己寫篇sql server觸發器的教程。 業務需求是這樣的,資料庫表中有兩個datetime 的欄位,StartTime、EndTime,當EndTime的值被插入或者修改時,自動計算Start ...
  • 作者:鬆軟科技(www.sysoft.net.cn) 發佈時間:2019/3/17 9:34:51 CREATE TABLE 語句 CREATE TABLE 語句用於創建資料庫中的表。 SQL CREATE TABLE 語法 CREATE TABLE 表名稱 ( 列名稱1 數據類型, 列名稱2 數據 ...
  • SQL SELECT INTO 語句可用於創建表的備份復件。 SELECT INTO 語句 SELECT INTO 語句從一個表中選取數據,然後把數據插入另一個表中。 SELECT INTO 語句常用於創建表的備份復件或者用於對記錄進行存檔。 SQL SELECT INTO 語法 您可以把所有的列插 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:本文代碼示例演示瞭如何在WPF中使用LiveCharts庫創建動態條形圖。通過創建數據模型、ViewModel和在XAML中使用`CartesianChart`控制項,你可以輕鬆實現圖表的數據綁定和動態更新。我將通過清晰的步驟指南包括詳細的中文註釋,幫助你快速理解並應用這一功能。 先上效果: 在 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • 概述:本示例演示了在WPF應用程式中實現多語言支持的詳細步驟。通過資源字典和數據綁定,以及使用語言管理器類,應用程式能夠在運行時動態切換語言。這種方法使得多語言支持更加靈活,便於維護,同時提供清晰的代碼結構。 在WPF中實現多語言的一種常見方法是使用資源字典和數據綁定。以下是一個詳細的步驟和示例源代 ...
  • 描述(做一個簡單的記錄): 事件(event)的本質是一個委托;(聲明一個事件: public event TestDelegate eventTest;) 委托(delegate)可以理解為一個符合某種簽名的方法類型;比如:TestDelegate委托的返回數據類型為string,參數為 int和 ...
  • 1、AOT適合場景 Aot適合工具類型的項目使用,優點禁止反編 ,第一次啟動快,業務型項目或者反射多的項目不適合用AOT AOT更新記錄: 實實在在經過實踐的AOT ORM 5.1.4.117 +支持AOT 5.1.4.123 +支持CodeFirst和非同步方法 5.1.4.129-preview1 ...
  • 總說周知,UWP 是運行在沙盒裡面的,所有許可權都有嚴格限制,和沙盒外交互也需要特殊的通道,所以從根本杜絕了 UWP 毒瘤的存在。但是實際上 UWP 只是一個應用模型,本身是沒有什麼許可權管理的,許可權管理全靠 App Container 沙盒控制,如果我們脫離了這個沙盒,UWP 就會放飛自我了。那麼有沒... ...
  • 目錄條款17:讓介面容易被正確使用,不易被誤用(Make interfaces easy to use correctly and hard to use incorrectly)限制類型和值規定能做和不能做的事提供行為一致的介面條款19:設計class猶如設計type(Treat class de ...
  • title: 從零開始:Django項目的創建與配置指南 date: 2024/5/2 18:29:33 updated: 2024/5/2 18:29:33 categories: 後端開發 tags: Django WebDev Python ORM Security Deployment Op ...
  • 1、BOM對象 BOM:Broswer object model,即瀏覽器提供我們開發者在javascript用於操作瀏覽器的對象。 1.1、window對象 視窗方法 // BOM Browser object model 瀏覽器對象模型 // js中最大的一個對象.整個瀏覽器視窗出現的所有東西都 ...