C++ 第三階段:語言改進 - 第四節:nullptr vs NULL

目錄

一、背景與概述

二、NULL?的定義與問題

1.?NULL?的定義

2.?NULL?的問題

三、nullptr?的定義與優勢

1.?nullptr?的定義

2.?nullptr?的優勢

四、nullptr?與?NULL?的對比

五、實際應用場景

1.?初始化指針

2.?函數調用與重載

3.?條件判斷

4.?模板與泛型編程

六、現代 C++ 的最佳實踐

1.?優先使用?nullptr

2.?避免?NULL?的隱式轉換

3.?兼容性處理

4.?統一代碼風格

七、總結

八、示例代碼

示例 1:nullptr?與?NULL?的區別

示例 2:nullptr?的類型安全性

示例 3:nullptr?在模板中的使用

C++從入門到入土學習導航_c++學習進程-CSDN博客


一、背景與概述

在 C++ 中,空指針的表示經歷了從 NULLnullptr 的演變。NULL 是 C 語言遺留下來的宏定義,而 nullptr 是 C++11 引入的專門用于空指針的關鍵字。兩者的區別主要體現在 類型安全函數重載可讀性現代 C++ 實踐 等方面。


二、NULL?的定義與問題

1.?NULL?的定義

  • NULL?是一個宏,通常定義為?0(在 C++ 中)或?(void*)0(在 C 中),例如:
    #define NULL 0       // C++ 中常見定義
    #define NULL (void*)0 // C 中常見定義
  • 由于?NULL?是宏,其本質是一個整數常量(0)或無類型指針((void*)0)。

2.?NULL?的問題

(1)?類型不安全
  • NULL?可以隱式轉換為任何整數類型,可能導致意外行為:
    int i = NULL; // 合法,但語義不明確
  • 與指針類型混淆,可能導致函數重載歧義(見下文)。
(2)?函數重載歧義
  • 如果存在兩個重載函數,一個接受整數參數,一個接受指針參數,使用?NULL?會調用整數版本:
    void func(int);   // 重載1
    void func(char*); // 重載2func(NULL); // 調用 func(int),因為 NULL 是 0
(3)?可讀性差
  • NULL?的語義不明確,容易讓開發者誤認為它是一個整數而不是空指針。

三、nullptr?的定義與優勢

1.?nullptr?的定義

  • nullptr?是 C++11 引入的關鍵字,表示空指針,其類型為?std::nullptr_t
  • 它是一個專門用于表示空指針的右值常量,不能隱式轉換為整數類型

2.?nullptr?的優勢

(1)?類型安全
  • nullptr?僅能賦值給指針類型,不會與整數混淆:
    int* p = nullptr; // 合法
    int i = nullptr;  // 錯誤:無法將 std::nullptr_t 轉換為 int
(2)?解決函數重載歧義
  • nullptr?會明確匹配指針類型的函數重載:
    void func(int);   // 重載1
    void func(char*); // 重載2func(nullptr); // 調用 func(char*),匹配指針類型
(3)?提升代碼可讀性
  • int* p = nullptr; // 語義明確:p 是一個空指針
(4)?兼容性與泛型編程
  • nullptr?的類型?std::nullptr_t?可以隱式轉換為任意指針類型,但在泛型編程中表現更穩定,減少類型推導錯誤。

四、nullptr?與?NULL?的對比

特性nullptrNULL
類型std::nullptr_t宏,通常為?0(C++)或?(void*)0(C)
類型安全性? 僅能賦值給指針類型? 可隱式轉換為整數或其他類型
函數重載匹配? 明確匹配指針類型? 可能匹配整數類型
可讀性? 語義明確? 可能引起歧義
推薦使用場景? C++11 及以上版本? 舊代碼或 C 兼容性需求
隱式轉換限制? 不能隱式轉換為整數? 可隱式轉換為整數
C 兼容性? C 不支持(C23 引入)? C 語言兼容

五、實際應用場景

1.?初始化指針

int* p1 = nullptr; // 推薦:C++11 及以上
int* p2 = NULL;    // 不推薦:舊代碼或 C 兼容性

2.?函數調用與重載

void func(int);
void func(char*);func(nullptr); // 調用 func(char*)
func(NULL);    // 調用 func(int)

3.?條件判斷

if (p == nullptr) {// 推薦:明確判斷指針是否為空
}
if (p == NULL) {// 不推薦:語義不明確
}

4.?模板與泛型編程

template <typename T>
void process(T* ptr) {if (ptr == nullptr) {// 安全處理空指針}
}

六、現代 C++ 的最佳實踐

1.?優先使用?nullptr

  • 在 C++11 及以上版本中,始終使用?nullptr?替代?NULL
  • nullptr?提供了更強的類型安全性和代碼可讀性。

2.?避免?NULL?的隱式轉換

  • 避免將?NULL?賦值給非指針類型(如?int),否則會導致編譯錯誤或運行時錯誤。

3.?兼容性處理

  • 在需要兼容舊代碼或與 C 語言交互時,可以保留?NULL,但應逐步遷移到?nullptr

4.?統一代碼風格

  • 在團隊或項目中統一使用?nullptr,減少因?NULL?導致的潛在問題。

七、總結

  • nullptr?是 C++11 引入的現代空指針表示方式,解決了?NULL?在類型安全、函數重載和可讀性上的問題。
  • 推薦始終使用?nullptr,除非必須兼容舊代碼或與 C 語言交互。
  • 通過使用?nullptr,可以編寫更安全、更清晰、更符合現代 C++ 標準的代碼。

八、示例代碼

示例 1:nullptr?與?NULL?的區別

#include <iostream>void func(int) {std::cout << "func(int)" << std::endl;
}void func(char*) {std::cout << "func(char*)" << std::endl;
}int main() {func(NULL);    // 輸出: func(int)func(nullptr); // 輸出: func(char*)return 0;
}

示例 2:nullptr?的類型安全性

#include <iostream>int main() {int* p1 = nullptr; // 合法int i = nullptr;   // 錯誤:無法將 std::nullptr_t 轉換為 intreturn 0;
}

示例 3:nullptr?在模板中的使用

#include <iostream>template <typename T>
void process(T* ptr) {if (ptr == nullptr) {std::cout << "Pointer is null" << std::endl;}
}int main() {int* p = nullptr;process(p); // 輸出: Pointer is nullreturn 0;
}

通過掌握 nullptrNULL 的區別,開發者可以編寫更安全、更高效的現代 C++ 代碼,避免因空指針導致的潛在問題。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/86471.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/86471.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/86471.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

計算機存儲器容量擴展設計實例解析

存儲器容量擴充是《計算機組成原理》課程的重要知識點。講解一個例題&#xff0c;以說明進行存儲器容量擴充設計的方法。 題目&#xff1a;在32位計算機系統中&#xff0c;用8K16位的SRAM芯片組成一個64KB的存儲器&#xff0c;已知起始地址為&#xff1a;6000 0000H。已知&…

轉載-秒殺系統—1.架構設計和方案簡介

轉載&#xff1a; https://mp.weixin.qq.com/s?__bizMzg5MzY5NDM3MQ&mid2247490866&idx1&sn0081517454680c85e0ed23eda4e82df5&chksmc02ba5fef75c2ce8b0c7f54182f3bda539230c75d2d75ed2b514b93decc0ff0c5de548a35dc3&cur_album_id3548464749150224391&…

Kubernetes中的容器生命周期回調

在介紹Kubernetes容器生命周期回調前&#xff0c;展示一個案例。 有個私有化部署的項目需要跑一個redis用作緩存&#xff0c;因redis中的數據不需要持久化&#xff0c;選擇在Kubernetes中通過deployment的方式部署&#xff0c;下面是deployment的代碼片段&#xff0c; ......…

基于STM32的工業倉庫環境智能監控系統設計

文章目錄 一、前言1.1 項目介紹【1】項目開發背景【2】設計實現的功能【3】項目硬件模塊組成【4】設計意義【5】市面上同類產品研究現狀【6】摘要 1.2 設計思路1.3 系統功能總結1.4 開發工具的選擇【1】設備端開發【2】上位機開發 1.5 模塊的技術詳情介紹【1】ESP8266-WIFI模塊…

如何在 Manjaro Linux 上啟用 AUR 倉庫來安裝軟件包

Manjaro 是基于 Arch 的系統&#xff0c;是了解和學習 Arch Linux 命令的絕佳方式。它自帶所有流行的桌面環境界面&#xff0c;無論是 XFCE 還是 Gnome 的愛好者&#xff0c;都可以在 Manjaro 中直接使用。 Manjaro 或 Arch Linux 的默認軟件包管理器是 Pacman&#xff0c;我們…

有限上升時間信號的反射波形

有限上升時間信號的反射波形: 從上一節討論中我們知道&#xff0c;阻抗不連續的點處&#xff0c;反射信號是入射信號的一個副本&#xff0c;并討論了上升時間為0的信號的反射情況。這些規律對于上升時間不為0的信號同樣適用&#xff0c;只不過入射信號和反射信號的疊加稍稍復雜…

Vue 3.4+ defineModel 全面詳解 + 實戰最佳實踐

&#x1f31f; 前言&#xff1a;為什么要關注 defineModel&#xff1f; 過去我們在 Vue 組件中使用 v-model 時&#xff0c;常需要這樣寫&#xff1a; // 子組件 defineProps([modelValue]) defineEmits([update:modelValue])function update(val) {emit(update:modelValue, …

MySQL事物隔離級別詳解

目錄 事物隔離級別總結 實際情況演示 臟讀&#xff08;未提交&#xff09; 避免臟讀&#xff08;讀已提交&#xff09; 不可重復讀 可重復讀 幻讀 事物隔離級別總結 SQL標準定義了四種事物隔離級別&#xff0c;用來平衡事物的隔離性&#xff08;Isolation&#xff09;和…

【安卓開發】Kotlin入門教程

一、Kotlin 基礎入門 1.1 Kotlin 簡介 Kotlin 是一種由 JetBrains 開發的靜態類型編程語言&#xff0c;運行在 Java 虛擬機上&#xff0c;也可以編譯為 JavaScript 或原生代碼。它于 2017 年被 Google 宣布為 Android 官方開發語言。 主要特點&#xff1a; 簡潔&#xff1a;…

工業機器人保護氣體節約方法

焊接在現代工業生產中作為一項關鍵技術&#xff0c;其效率和質量直接影響著產品的最終性能和生產成本。隨著智能制造的不斷推進&#xff0c;工業焊接機器人在自動化生產線中扮演著越來越重要的角色。焊接過程中的氣體調節一直是一個技術難題&#xff0c;它直接關系到焊接質量的…

java中集合API

集合API 一、簡述二、Collection&#xff08;1&#xff09;.List1.ArrayList&#xff1a;2. LinkedList &#xff08;2&#xff09; Set 三、Map1.HashMap2. TreeMap 三、Iterator接口Collections工具類1.對集合元素進行排序2.基于Comparator&#xff0c;自定義方法(內部類)進行…

Oracle 用戶權限與對象權限詳解

一、權限體系概述 Oracle 數據庫的權限管理是保障數據安全的核心機制&#xff0c;主要分為系統權限&#xff08;System Privileges&#xff09; 和對象權限&#xff08;Object Privileges&#xff09; 兩大類&#xff1a; 系統權限&#xff1a;賦予用戶在數據庫中執行特定操作…

kubectl get pods 時 “client-side throttling, not priority and fairness“ 原因初探

1. 問題起源 在kubernetes里&#xff0c;使用kubectl get pods 時&#xff0c;返回 I0508 05:43:04.655602 100742 request.go:668] Waited for 1.178494016s due to client-side throttling, not priority and fairness, request: GET:https://10.103.0.1:443/apis/cert-ma…

力扣網C語言編程題:位運算來解決 “尋找重復數”

一. 簡介 前面兩篇文章解決力扣網上"查找重復數"的題目&#xff0c;提供了三種思路&#xff1a;哈希表、二分法和快慢指針。文章如下&#xff1a; 力扣網C語言編程題&#xff1a;“尋找重復數”的兩種思路-CSDN博客 力扣網C語言編程題&#xff1a;快慢指針來解決 …

3D視覺感知

目錄 3D視覺感知任務 單目3D感知 單目3D物體檢測 – 直接預測3D信息 單目3D物體檢測 – 總結 單目深度估計 雙目3D感知 多目3D感知 3D視覺感知任務 ? 輸入&#xff1a;單攝像頭或多攝像頭生成的圖像數據 ? 單張圖像 ? 圖像序列 ? 輸出 ? 稀疏&#xff1a…

es中常規的根據字段查詢時走什么索引(說明:「常規的根據字段查詢」不包含分詞查詢)

在Elasticsearch中&#xff0c;“常規的根據字段查詢”且不涉及分詞的查詢&#xff08;如精確匹配、范圍查詢&#xff09;&#xff0c;主要依賴以下索引機制&#xff1a; 一、核心索引類型及適用場景 字段類型索引結構典型查詢方式應用場景keyword倒排索引&#xff08;未分詞…

MYSQL如何插入數據,效率會更高

在MySQL中&#xff0c;插入數據的效率可以通過多種方式逐步提升。以下是從簡單到復雜的優化路徑&#xff0c;幫助你逐步提高數據插入的性能&#xff1a; 一、基礎插入&#xff1a;逐條插入 這是最基礎的插入方式&#xff0c;適用于少量數據的插入操作。雖然簡單&#xff0c;但…

Rabbitmq的五種消息類型介紹,以及集成springboot的使用

交換機類型 Fanout Exchange 扇型交換機&#xff0c;這個交換機沒有路由鍵概念&#xff0c;就算你綁了路由鍵也是無視的。 這個交換機在接收到消息后&#xff0c;會直接轉發到綁定到它上面的所有隊列 Direct Exchange 直連型交換機&#xff0c;根據消息攜帶的路由鍵將消息投遞…

日語學習-日語知識點小記-進階-JLPT-真題訓練-N2階段(4):2022年12月2023年12月

日語學習-日語知識點小記-進階-JLPT-真題訓練-N2階段&#xff08;4&#xff09;&#xff1a;2022年12月&2023年12月 1、前言&#xff08;1&#xff09;情況說明&#xff08;2&#xff09;工程師的信仰&#xff08;3&#xff09;真題訓練 2、2個卷的單詞部分1、 真題-2023年…

從代碼學習深度強化學習 - Actor-Critic 算法 PyTorch版

文章目錄 前言算法原理1. 從策略梯度到Actor-Critic2. Actor 和 Critic 的角色3. Critic 的學習方式:時序差分 (TD)4. Actor 的學習方式:策略梯度5. 算法流程代碼實現1. 環境與工具函數2. 構建Actor-Critic智能體3. 組織訓練流程4. 主程序:啟動訓練5. 實驗結果總結前言 在深…