跟我学C++中级篇——空值的定义

news/2024/12/22 2:21:51/

一、空值

在提到c++/c的空值时,先扯远一些。谈一谈数学中的0,0的出现要晚于其它的数,而0的出现却引发了数学的极大的发展和进步。而在计算机科学中,在使用一个变量时,它的值的可能性有很多,其中,就有可能会是啥都没有。一定会有很多有经验的开发者明白,不给变量赋值并不代表变量没有值,而是这个值可能是0,也可能是随机值。这就有可能在第一次使用时,出现一些意外的情况。所以,变量一定要给予初始赋值几乎成为了一条开发者必须遵守的规则。
既然要赋值,那提到的啥都没有,怎么表示呢?对,就是空值。在数学中,0代表着没有值,这不正好就是空值么?所以就简装的移植到了计算机科学中。所以在C/C++中,0就代表了空值。为了更好的表示这个空值,给它取了一个别名NULL。

二、NULL和nullptr

先看一下在常用的情况下,如何使用空值,代码如下:

#define ABC NULL
int main()
{int * ptr = NULL;return 0;
}

是不是由此觉得不过而已,就是一个简单的值的定义么。OK,确实如此。但细节要出来了,先看一下NULL的定义:

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

上面的代表有两个问题,一是为什么NULL在不同的语言中定义不同;二是为什么要把指针定义为空值。后者比较好回答,指针的地址指向空(0),则表示进入了安全管理区域,直接在开发中应用,编译器会报一个异常出现,可以理解成类似于“陷阱”的东西。而前者说起来就比较麻烦了。
先从标准本身来看,早期的c++标准(03以前)规定空指针常量是一个整数类型的右值常量表达式,其结果为必须为0。同时要求其必须可以转换为相对应的指针类型,并可以与其它指针类型(如对象或函数指针)区别开来。然后它规定还必须可以判断两个同类型的空指针相等。而C的标准规定,值为0的常量表达式或者可以转换为void*的表达式称为空指针常量,而如果将空指针常量转换为指针类型,其生成的类型指针必须能与其它对象或函数指针不同。
所以说定义规定的不同,自然就会有编译器支持的相关不同。这个要理解明白。因此,在C和C++中处理NULL,就需要重定定义一下,否则可能出现编译的问题。
所有的开发者都明白的是,一个语言如果长期保持与其它语言的兼容,势必带来更多的技术负担。所以c++做为一个后来者,完全可以自行定义一个空指针,而不必强力要求必须与C兼容(当然说和做是不是一回事,编译器厂商也不是铁板一块)。这时,就提出了nullptr.
nullptr为什么会出现?主要有几个原因:
1、为了保持C++标准的独立性
2、为空值提供一个专有名字并成为一个保留的关键字
3、能够区别于C中这种整数与空的混用,即无法在算数表达式中运算
4、支持标准中的提到的可以转换成任何类型的指针类型而无法被转换为其它非指针类型
5、支持C++异常机制(指针类型),虽然不建议使用异常机制

三、NULL和nullptr的比较

0做为一个整数,又代表空值,这让开发者和维护者,更让编译器处理起来比较麻烦。他们需要根据不同的上下文来判断,这个0到底是整数0还是一个指针的空值。而将NULL定义为0,从上面的代码可以看出,它其实就是一个宏定义。
宏定义的缺点对开发者们并不陌生。在近些年的C/C++编程中,一个重要的推荐方法就是在程序中尽量避免使用宏。
另外,使用NULL时,如果单纯从后面应用很难简单的判断出其为一个整数还是一个指针。对于模糊的东西,不光是开发者,编译器也会进一步的增加判断条件,这样的结果其实是一个不友好的现象。也不符合设计原则中的单一职责,或许本来就应该是你是你我是我,而不需要再通过一个ID来区分你我的不同。这等于是徒耗精力和时间。
同样,对于这种定义不清晰的标准,不同的编译器厂商可能就会有不同的理解,那么产生的结果可能就有所不同。这也是为什么在Windows上编译没有问题的代码可能到Linux上编译就有问题(排除不同接口定义不同)。这对于开发者来说其实是一件很痛苦的事情,维护着不同的平台的相同功能的代码,很可能会导致一些意外的“惊喜”。
而如果引入nullptr,则不必再考虑这些问题,桥归桥路归路,大路朝天,各走一边。你的问题就是你的问题,不要找我的问题。界线清晰了,无论是开发者还是编译器其实都喜欢这种清爽的风格。

四、总结

后发语言往往能看到借鉴语言的缺点和优点。这和学习一样,有前人的经验会进步更快。所以,从这一个小的问题,可以引申出来很广大的思考。从细节的区分到设计模式、原则和设计思想的改变。这就是老师经常说的“把薄书读厚了”。


http://www.ppmy.cn/news/1536330.html

相关文章

Vue.js 事件处理器

1. 基本用法 在 Vue.js 中&#xff0c;事件处理器可以通过 v-on 指令来绑定。你可以使用简写形式 来简化代码。 <template><button click"handleClick">点击我</button> </template><script> export default {methods: {handleClic…

算法专题三: 二分查找

目录 1. 朴素版: 二分查找2. 查找排序数组元素第一个和最后一个位置3. 搜索插入位置4. x的平方根5. 山脉数组的峰顶索引6. 寻找旋转数组中的最小值7. 点名 博客主页: 酷酷学!!! 感谢您的关注~ 正文开始 1. 朴素版: 二分查找 题目思路: 仅需根据题意, 找出二段性, 正确更新下标…

实施威胁暴露管理、降低网络风险暴露的最佳实践

随着传统漏洞管理的发展&#xff0c;TEM 解决了因攻击面扩大和安全工具分散而产生的巨大风险。 主动式 TEM 方法优先考虑风险并与现有安全工具无缝集成&#xff0c;使组织能够在威胁被有效利用之前缓解威胁。 为什么威胁暴露管理 (TEM) 在现代网络安全策略中变得至关重要&…

力扣59.螺旋矩阵||

题目链接&#xff1a;59. 螺旋矩阵 II - 力扣&#xff08;LeetCode&#xff09; 给你一个正整数 n &#xff0c;生成一个包含 1 到 n2 所有元素&#xff0c;且元素按顺时针顺序螺旋排列的 n x n 正方形矩阵 matrix 。 示例 1&#xff1a; 输入&#xff1a;n 3 输出&#xff…

深度解析内网横向移动及防御策略

随着信息技术的飞速发展&#xff0c;企业内网已成为业务运营的核心支柱&#xff0c;承载着大量敏感数据和关键业务逻辑。然而&#xff0c;与此同时&#xff0c;网络攻击技术也在不断演进&#xff0c;特别是横向移动攻击&#xff0c;已成为高级威胁者渗透和实现远控的必要手段。…

vue3导入本地图片2种实现方法

在<script setup>中使用import语法&#xff1a; <template><img :src"logo" alt"Logo"> </template><script setup> import logo from ./assets/logo.png; </script> 使用Vue的ref来动态地在<script setup>中…

如何使用ssm实现基于Java的民宿预订管理系统的设计与实现

TOC ssm773基于Java的民宿预订管理系统的设计与实现jsp 绪论 1.1课题研究背景意义 随着科技的发展&#xff0c;计算机的应用&#xff0c;人们的生活方方面面都和互联网密不可分。计算机的普及使得人们的生活更加方便快捷&#xff0c;网络也遍及到我们生活的每个角落&#x…

鹏哥C语言62---第9次作业:函数递归练习

#define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <string.h> //-------------------------------------------------------------------------------------------第九次作业 函数递归等 //-----------------------------------------------------…