【C++掌中宝】C++ 中的空指针救世主——nullptr

在这里插入图片描述

文章目录

  • 1. 什么是 NULL?
  • 2. NULL 在 C 和 C++ 中的区别
  • 3. C++11 引入 nullptr 的原因
  • 4. nullptr 与 NULL 的区别
  • 5. nullptr 的应用场景
  • 6. 模拟 nullptr 的实现
  • 7. 总结
  • 结语

1. 什么是 NULL?

在 C 和 C++ 编程中,NULL 常用于表示空指针,但它本质上是一个宏定义

  • 在C语言中,NULL 通常被定义为 (void *)0,也就是一个指向空的 void 指针。
  • 但在 C++ 中,NULL 常常被定义为整数常量 0

在传统的C头文件(stddef.h)中,可以看到如下代码:

#ifndef NULL#ifdef __cplusplus#define NULL 0#else#define NULL ((void *)0)#endif
#endif

由于在 C++ 中,void* 不能隐式转换为其他类型的指针,因此将 NULL 定义为 0。这导致在函数重载时可能会出现二义性问题

2. NULL 在 C 和 C++ 中的区别

在 C 语言中,NULL 定义为 (void *)0,它可以隐式转换为任何类型的指针。这使得以下代码在 C 中可以正常编译和运行:

int *pi = NULL;
char *pc = NULL;

然而,在 C++ 中,由于更严格的类型检查,void* 不能隐式转换为其他指针类型。因此,C++ 中的 NULL 通常被定义为整数常量 0。这导致了以下代码在 C++ 中的编译错误

int *pi = (void*)0;  // OK in C, error in C++

报错如下

在这里插入图片描述


除此之外,NULL 在函数重载时也会导致意外行为。例如,考虑以下代码:

#include<iostream>
using namespace std;
void f(int x)
{cout << "f(int x)" << endl;
}
void f(int* ptr)
{cout << "f(int* ptr)" << endl;
}
int main()
{f(0);f(NULL);
} // 这会调用哪个版本的 f()?

运行结果如下
在这里插入图片描述

由于 NULL 被定义为 0,这里会调用 int 参数的版本,而不是我们期望的指针版本,所以最后两个结果都是选择int 参数的版本。

3. C++11 引入 nullptr 的原因

为了消除 NULL 在 C++ 中的二义性问题,C++11 引入了 nullptr 作为一种新型空指针常量nullptr 是一个关键字,代表一种特殊类型,使用nullptr定义空指针可以避免类型转换的问题,因为它只能转换为任意指针类型,但不能转换为整数类型

例如,以下代码可以正确运行并调用我们期望的函数版本:

#include<iostream>
using namespace std;void f(int x) {cout << "foo(int x)" << endl;
}void f(int* ptr) {cout << "foo(int* ptr)" << endl;
}int main() {f(nullptr);  // 调用 f(int* ptr)return 0;
}

nullptr 的引入解决了函数重载中的二义性问题,确保空指针总是正确地匹配到指针类型的重载函数。

4. nullptr 与 NULL 的区别

虽然 NULL 仍然可以在 C++ 中使用,但它在特定场景下会引发错误或误解。相比之下,nullptr 是一种更加安全且明确的选择:

  • 在C++中NULL 通常是整数 0,会引发二义性问题。
  • nullptr 是一个指针常量,可以隐式转换为任意指针类型,不会引发二义性问题。

例如:

void func(int x) {cout << "func(int)" << endl;
}void func(char* p) {cout << "func(char*)" << endl;
}int main() {func(0);         // 调用 func(int)func(nullptr);   // 调用 func(char*)return 0;
}

5. nullptr 的应用场景

在 C++11 及更高版本中,推荐使用 nullptr 代替 NULL 来表示空指针,特别是在函数重载的场景下。例如:

int* p1 = nullptr;
if (p1 == nullptr) {cout << "p1 是空指针" << endl;
}

需要注意的是,nullptr 不能赋值给整数类型,因此以下代码会导致编译错误:

int n = nullptr;  // 编译错误

报错如下
在这里插入图片描述

6. 模拟 nullptr 的实现

对于不支持 C++11 的编译器,我们可以自己模拟实现 nullptr。以下是一个简单的实现:

const class nullptr_t {
public:template<class T>operator T*() const { return 0; }template<class C, class T>operator T C::*() const { return 0; }private:void operator&() const;
} nullptr = {};

7. 总结

NULL 是一个历史遗留的宏定义,虽然在 C++ 中仍然可以使用,但它会在特定场景下引发意外的行为和错误。为了消除这些问题,C++11 引入了 nullptr,作为一种类型安全的空指针表示。

在现代 C++ 编程中,强烈推荐使用 nullptr 代替 NULL,确保代码在空指针处理上更加明确和安全。

结语

通过本文,读者应当已经掌握了 nullptr 的基本概念、与 NULL 的区别以及如何在实际编程中应用它。如果你还在用 NULL,是时候拥抱 nullptr,让你的代码更加健壮吧!

今天的分享到这里就结束啦!如果觉得文章还不错的话,可以三连支持一下。

也可以点点关注,避免以后找不到我哦!

Crossoads主页还有很多有趣的文章,欢迎小伙伴们前去点评,您的支持就是作者前进的动力!
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/431472.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何修改音频的音量增益

一、前言 在开发音频相关的功能&#xff08;比如说语音通话、播放音乐&#xff09;时&#xff0c;经常会遇到音量太小的问题&#xff0c;这时候就需要我们对原始数据进行处理。本文将介绍如何通过修改原始音频数据来增加增益&#xff0c;本文包含如下内容&#xff1a; 1.音频数…

【Java】虚拟机(JVM)内存模型全解析

目录 一、运行时数据区域划分 版本的差异&#xff1a; 二、程序计数器 程序计数器主要作用 三、Java虚拟机 1. 虚拟机运行原理 2. 活动栈被弹出的方式 3. 虚拟机栈可能产生的错误 4. 虚拟机栈的大小 四、本地方法栈 五、堆 1. 堆区的组成&#xff1a;新生代老生代 …

速通LLaMA3:《The Llama 3 Herd of Models》全文解读

文章目录 概览论文开篇IntroductionGeneral OverviewPre-TrainingPre-Training DataModel ArchitectureInfrastructure, Scaling, and EfficiencyTraining Recipe Post-TrainingResultsVision ExperimentsSpeech Experiments⭐Related WorkConclusionLlama 3 模型中的数学原理1…

目标检测系列(一)什么是目标检测

目录 一、相关名词解释 二、目标检测算法 三、目标检测模型 四、目标检测应用 五、目标检测数据集 六、目标检测常用标注工具 一、相关名词解释 关于图像识别的计算机视觉四大类任务&#xff1a; 分类&#xff08;Classification&#xff09;&#xff1a;解决“是什么&…

【LLM开源项目】LLMs-微调框架-LLaMA-Factory入门指南v3.0

【#】SFT 训练 项目地址&#xff1a;https://llamafactory.readthedocs.io/zh-cn/latest/getting_started/sft.html 可以使用以下命令进行微调&#xff1a; llamafactory-cli train examples/train_lora/llama3_lora_sft.yamlexamples/train_lora/llama3_lora_sft.yaml 提供…

车辆识别数据集,图片数量20500,模型已训练200轮

车辆识别数据集&#xff08;Vehicle Recognition Dataset, VDRD&#xff09; 摘要 VDRD 是一个专为车辆识别设计的大规模数据集&#xff0c;它包含了20500张不同类型的汽车、货车、公交车以及其他类型车辆的图像。数据集提供了四种车辆类别&#xff1a;汽车、货车、其他车辆和…

单片机原理及应用

单片机原理 一、单片机概述1.1 各种单片机系列1.2嵌入式处理器1.2.1 嵌入式DSP1.2.2 嵌入式微处理器 二、AT89S522.1 硬件组成2.2 引脚功能2.2.1 电源、时钟引脚2.2.2 控制引脚2.2.3 并行I/O口引脚 2.3 AT89S52单片机的CPU2.3.1 运算器2.3.2 控制器 2.4 AT89S52单片机的存储器结…

【代码随想录训练营第42期 Day60打卡 - 图论Part10 - Bellman_ford算法系列运用

目录 一、Bellman_ford算法的应用 二、题目与题解 题目一&#xff1a;卡码网 94. 城市间货物运输 I 题目链接 题解&#xff1a;队列优化Bellman-Ford算法&#xff08;SPFA&#xff09; 题目二&#xff1a;卡码网 95. 城市间货物运输 II 题目链接 题解&#xff1a; 队列优…

MySQL_表_进阶(1/2)

我们的进阶篇中&#xff0c;还是借四张表&#xff0c;来学习接下来最后关于表的需求&#xff0c;以此完成对表的基本学习。 照例给出四张表&#xff1a; 学院表&#xff1a;(testdb.dept) 课程表&#xff1a;(testdb.course) 选课表:&#xff08;testdb.sc&#xff09; 学生表…

python调用c++动态链接库,环境是VS2022和vscode2023

目录 前言&#xff1a;配置环境&#xff1a;基础夯实&#xff08;对于ctypes的介绍&#xff09;&#xff1a;1. 加载共享库2. 定义函数原型3. 调用函数4. 处理数据结构5. 处理指针6. 错误处理7. 使用 ctypes.util总结 效果展示&#xff1a;操作步骤(保姆级教学)一在VS中创建dll…

YOLOv8——测量高速公路上汽车的速度

引言 在人工神经网络和计算机视觉领域&#xff0c;目标识别和跟踪是非常重要的技术&#xff0c;它们可以应用于无数的项目中&#xff0c;其中许多可能不是很明显&#xff0c;比如使用这些算法来测量距离或对象的速度。 测量汽车速度基本步骤如下&#xff1a; 视频采集&#x…

(done) 声音信号处理基础知识(4) (Understanding Audio Signals for ML)

来源&#xff1a;https://www.youtube.com/watch?vdaB9naGBVv4 模拟信号特点如下 时域连续(x轴) 振幅连续(y轴) 如下是模拟信号的一个例子&#xff1a; 数字信号特点如下&#xff1a; 一个离散值序列 数据点的值域是一系列有限的值 ADC&#xff1a;模拟信号到数字信号的…

python全栈学习记录(十八)re、os和sys、subprocess

re、os和sys、subprocess 文章目录 re、os和sys、subprocess一、re1.正则字符2.正则表达式的使用3.group的使用4.贪婪匹配与惰性匹配5.其他注意事项 二、os和sys1.os2.sys 三、subprocess四、打印进度条 一、re python中的re模块用来使用正则表达式&#xff0c;正则就是用一系…

【Python机器学习】NLP信息提取——提取人物/事物关系

目录 词性标注 实体名称标准化 实体关系标准化和提取 单词模式 文本分割 断句 断句的方式 使用正则表达式进行断句 词性标注 词性&#xff08;POS&#xff09;标注可以使用语言模型来完成&#xff0c;这个语言模型包含词及其所有可能词性组成的字典。然后&#xff0c;该…

http增删改查四种请求方式操纵数据库

注意&#xff1a;在manage.py项目入口文件中的路由配置里&#xff0c;返回响应的 return语句后面的代码不会执行&#xff0c;所以路由配置中每个模块代码要想都执行&#xff0c;不能出现return 激活虚拟环境&#xff1a;venv(我的虚拟环境名称&#xff09;\Scripts\activate …

java项目发布后到Tomcat时,总是带一层路径解决方案

java项目发布后到Tomcat时,总是带一层路径 参考文章&#xff1a;java 线上项目访问项目 会多一层项目根路径 根据参考文章写的这篇文章&#xff0c;部分文章细节有完善和改动 在Java Web应用中&#xff0c;当你把应用发布到Tomcat时&#xff0c;如果应用的web.xml配置文件中的&…

Karmada新版本发布,支持联邦应用跨集群滚动升级

摘要&#xff1a;本次升级支持联邦应用跨集群滚动升级&#xff0c;使用户版本发布流程更加灵活可控&#xff1b;透明同事karmadactl 新增了多项运维能力&#xff0c;提供独特的多集群运维体验。 本文分享自华为云社区 《Karmada v1.11 版本发布&#xff01;新增应用跨集群滚动升…

柔性数组 初学版

1.定义 结构中的最后⼀个元素允许是未知⼤⼩的数组&#xff0c;这就叫做『柔性数组』成员 有些编译器会报错⽆法编译可以改成&#xff1a; typedef struct st_type { int i; int a[]; // 柔性数组成员 }type_a; 2.柔性数组的特点&#xff1a; • 结构中的柔性数组成员前…

ReadWriteLock读写锁

读写锁基本概念 ReadWriteLock是Java并发包中的一个接口&#xff0c;它定义了两种锁&#xff1a;读锁&#xff08;Read Lock&#xff09;和写锁&#xff08;Write Lock&#xff09;&#xff0c;真正的实现类是ReentrantReadWriteLock。读锁允许多个线程同时读取共享资源&#…

JAVA开源项目 体育馆管理系统 计算机毕业设计

本文项目编号 T 048 &#xff0c;文末自助获取源码 \color{red}{T048&#xff0c;文末自助获取源码} T048&#xff0c;文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析5.4 用例设计 六、核…