C++深入探究用NULL来初始化空指针是否合适

目录
  • 1.C++98中的空指针
    • 指针的危险性
  • 2.C++11中的空指针

我们要了解C++11新特性的nullptr,我们很有必要先了解一下C++11之前的程序员是怎么使用空指针的。

1.C++98中的空指针

我们知道,在良好的C/C++编程习惯中,声明一个变量时最好给这个变量赋一个合适的初始值,否则就有可能出现不可预料的错误。

指针的危险性

在C++中创建指针时,计算机将分配用来存储地址的内存,但不会分配用来存储指针所指向的的数据的内存。为数据提供空间的是一个独立的步骤,忽略这一步无疑是自找麻烦的,如下所示:

int* fellow;
*fellow = 1234;

fellow确实是一个指针,但它指向哪里呢?上述代码没有将地址赋给fellow,那么1234将被放置在哪个内存单元呢?我们并不知道。有fellow没有被初始化,他可能有任何值。不管值是什么,程序都将他解释为存储1234的地址。但是如果fellow的值碰巧为1000,计算机将把数据放在地址1000上,即使这恰巧是程序代码的地址,fellow指向的地方很可能并不是所要存储的1234的地方,这种错误可能会导致一些最隐匿,最难以跟踪的bug。

因此为了避免这个问题,我们都要给指针进行初始化。通常我们都是这样来初始化指针的。

int main()
{
	//空指针定义
	int* p1 = NULL;
	int* p2 = 0;
	return 0;
}

实际上NULL是一个宏,我们在传统的C头文件(stddef.h)中就可以看到如下代码:

可以看到, NULL 可能被定义为字面常量 0 ,或者被定义为无类型指针 (void*) 的常量 。不论采取何种定义,在使用空值的指针时,都不可避免的会遇到一些麻烦。 比如:下面这段代码的输出结果是什么?

void f(int)
{
	cout << "f(int)" << endl;
}
void f(int*)
{
	cout << "f(int*)" << endl;
}
int main()
{
	//空指针定义
	int* p1 = NULL;
	int* p2 = 0;
	f(0);
	f(NULL);
	return 0;
}

按照我们正常的想法f(0)应该进入void f (int),f(NULL)进入void f(int*),因此我们想要得到的结果是分别打印f(int)和f(int*)。我们来看运行结果是否和我们所想的一样。

我们发现运行结果和我们所想存在出入,这是为什么?

这是因为程序本意是想通过f(NULL)调用指针版本的f(int*)函数,但是由于NULL被宏定义成了0,在预处理阶段,NULL已经被宏替换成了0,因此f(NULL)函数就已经被替换成了f(0),因此我们才会得到两个相同的打印结果。

在 C++98 中,字面常量 0 既可以是一个整形数字,也可以是无类型的指针 (void*) 常量,但是编译器默认情况下将其看成是一个整形常量,如果要将其按照指针方式来使用,必须对其进行强转(void *)0 。

2.C++11中的空指针

为了避免上述这个问题,C++11引入了nullptr关键字来表示指针空值。

int main()
{
	//空指针定义
	int* p1 = NULL;
	int* p2 = 0;
	//推荐
	int* p3 = nullptr;
	f(0);
	f(NULL);
	f(nullptr);
	return 0;
}

因此我们再次传入nullptr,看看他的结果是什么?

此时我们发现nullptr解决了这个问题。

注意:

1. 在使用 nullptr 表示指针空值时,不需要包含头文件,因为 nullptr 是 C++11 作为新关键字引入的 。

2. 在 C++11 中, sizeof(nullptr) 与 sizeof((void*)0) 所占的字节数相同。 在32位机器下大小为4,在64位下大小为8。

3. 为了提高代码的健壮性,在后续表示指针空值时建议最好使用 nullptr 。

结论:nullptr是对NULL的一个升级,因此在以后的初始化空指针时,建议大家使用nullptr。

到此这篇关于C++深入探究用NULL来初始化空指针是否合适的文章就介绍到这了,更多相关C++ 空指针初始化内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C++11 nullptr实现初始化空指针

    实际开发中,避免产生“野指针”最有效的方法,就是在定义指针的同时完成初始化操作,即便该指针的指向尚未明确,也要将其初始化为空指针. 所谓“野指针”,又称“悬挂指针”,指的是没有明确指向的指针.野指针往往指向的是那些不可用的内存区域,这就意味着像操作普通指针那样使用野指针(例如 &p),极可能导致程序发生异常. C++98/03 标准中,将一个指针初始化为空指针的方式有 2 种: int *p = 0; int *p = NULL; //推荐使用 可以看到,我们可以将指针明确指向 0(0x0000

  • C++深入探究用NULL来初始化空指针是否合适

    目录 1.C++98中的空指针 指针的危险性 2.C++11中的空指针 我们要了解C++11新特性的nullptr,我们很有必要先了解一下C++11之前的程序员是怎么使用空指针的. 1.C++98中的空指针 我们知道,在良好的C/C++编程习惯中,声明一个变量时最好给这个变量赋一个合适的初始值,否则就有可能出现不可预料的错误. 指针的危险性 在C++中创建指针时,计算机将分配用来存储地址的内存,但不会分配用来存储指针所指向的的数据的内存.为数据提供空间的是一个独立的步骤,忽略这一步无疑是自找麻烦

  • C++11中longlong超长整型和nullptr初始化空指针

    目录 1. C++11:long long 超长整型 2. C++11:nullptr 初始化空指针 本文介绍 C++11 标准中新添加的 long long 超长整型和 nullptr 初始化空指针. 1. C++11:long long 超长整型 C++ 11 标准中,基于整数大小的考虑,共提供了如下表所示的这些数据类型.与此同时,标准中还明确限定了各个数据类型最少占用的位数. 整数类型 等价类型 C++11标准规定占用最少位数 short short int(有符号短整型) 至少 16 位

  • 解析C语言中空指针、空指针常量、NULL & 0的详解

    什么是空指针常量(null pointer constant)?[6.3.2.3-3] An integer constant expression with the value 0, or such an expression cast to type void *, is called a null pointer constant. 这里告诉我们:0.0L.'\0'.3 - 3.0 * 17 (它们都是"integer constant expression")以及 (void*

  • 详解Kotlin的空指针处理

    详解Kotlin的空指针处理 Kotlin的空指针处理相比于java有着极大的提高,可以说是不用担心出现NullPointerException的错误,kotlin对于对象为null的情况有严格的界定,编码的阶段就需要用代码表明引用是否可以为null,为null的情况需要强制性的判断处理. 咋看一下这些在java里面其实也有,问题是一般开发中不写也是可以的(大部分开发不会花很多时间考虑这些),等出了空指针错误再一个个打补丁.这样往往会遗漏很多空指针,后期的解决仅仅是做一个if判断,没有从根源解决

  • 详解Spring 中如何控制2个bean中的初始化顺序

    开发过程中有这样一个场景,2个 bean 初始化逻辑中有依赖关系,需要控制二者的初始化顺序.实现方式可以有多种,本文结合目前对 Spring 的理解,尝试列出几种思路. 场景 假设A,B两个 bean 都需要在初始化的时候从本地磁盘读取文件,其中B加载的文件,依赖A中加载的全局配置文件中配置的路径,所以需要A先于B初始化,此外A中的配置改变后也需要触发B的重新加载逻辑,所以A,B需要注入彼此. 对于下面的模型,问题简化为:我们需要initA()先于initB()得到执行. @Service pu

  • Oracle ORA-22908(NULL表值的参考)异常分析与解决方法

    场景如下: --创建类型(type)create or replace type list_obj is table of number;--创建表结构create table test( name varchar2(30) primary key check(regexp_like(name,'^([a-z]|[0-9]|_)+$')), id number not null, version_list list_obj) nested table version_list store as

  • C++ this指针和空指针的具体使用

    每一个非静态成员函数只会诞生一份函数实例,也就是说多个同类型的对象会共用一块代码那么问题是:这-块代码是如何区分那个对象调用自己的呢? C++通过提供特殊的对象指针,this指针,解决上述问题.this指针指向被调用的成员函数所属的对象 this指针是隐含每一个非静态成员函数内的-种指针 this指针不需要定义,直接使用即可 this指针的用途: 当形参和成员变量同名时,可用this指针来区分 在类的非静态成员函数中返回对象本身,可使用return this 一.this指针 1 this指针解

  • C++中NULL与nullptr的区别对比

    前言 在编写C程序的时候只看到过NULL,而在C++的编程中,我们可以看到NULL和nullptr两种关键字,其实nullptr是C++11版本中新加入的,它的出现是为了解决NULL表示空指针在C++中具有二义性的问题,为了弄明白这个问题,我查找了一些资料,总结如下. 一.C程序中的NULL 在C语言中,NULL通常被定义为:#define NULL ((void *)0) 所以说NULL实际上是一个空指针,如果在C语言中写入以下代码,编译是没有问题的,因为在C语言中把空指针赋给int和char

  • Mybatis初始化知识小结

    目录 一.前言 二.MyBatis的初始化做了什么 2.1 Mybatis的初始化过程就是加载自己运行时所需要的配置信息的过程 2.2 Mybatis的配置信息有哪些 2.3 mybatis-config.xml与Configuration类 2.4 MyBatis初始化的两种方式 三.MyBatis基于XML配置文件创建Configuration对象的过程 3.1 定位到Mybatis初始化的关键一句 3.2 MyBatis初始化基本过程 3.2.1 Mybatis初始化时序图 3.2.2 M

随机推荐