C++11之空指针-nullptr
Posted 林夕07
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++11之空指针-nullptr相关的知识,希望对你有一定的参考价值。
系列文章
C++11之正则表达式(regex_match、regex_search、regex_replace)
C++11之线程库(Thread、Mutex、atomic、lock_guard、同步)
C++11之智能指针(unique_ptr、shared_ptr、weak_ptr、auto_ptr)浅谈内存管理
C++11之强制类型转换(static_cast,const_cast,dynamic_cast,reinterpret_cast)
C++11之decltype类型推导(使用场景、推导四规则、cv限定符)
目录
空指针:从0到NULL再到nullptr历程
在C/C++早期对指针初始化时,都是赋予0
或者NULL
值。但是计算机0
地址的内存空间往往是不能被修改的,所以在代码如果直接赋值的话,程序就直接崩溃了。
在库中可以看到C++中NULL
是0
,C中NULL
是(void)0
。无论是那种定义都会导致一些意外。
#ifndef NULL
#ifdef __cplusplus
#define NULL 0
#else
#define NULL ((void *)0)
#endif
#endif
具体我们来看一个例子,在下面这个代码中,我们重载了f函数,并且尝试通过传入参数NULL调用指针版本的函数,结果恰恰相反。因为NULL
会被翻译为0
,所以和调用0
当作参数的结果是相同的。
#include <iostream>
void f(char* c)
std::cout << "invoke f(char* c)" << std::endl;
void f(int i)
std::cout << "invoke f(int i)" << std::endl;
int main()
f(0);
f(NULL);
f((char*)0);
return 0;
运行结果:
invoke f(int i)
invoke f(int i)
invoke f(char* c)
引起这些问题的本质在于字面常量0具有二义性,在C++98标准中,字面常量
0
的类型既可以是一个整型,还可以是一个无类型的指针(void*)
。如果想将字面常量0
当作指针那么就需要C风格的强制类型转换(char*)0
。
nullptr与nullptr_t
C++11引入了nullptr和nullptr_t,一般来说nullptr就够了,nullptr_t目前作用不大(俩者等价),用法非常简单,原本的NULL替换为nullptr即可。
在标准中规定了一些规则:
- 所有定义为nullptr_t类型的数据都是等价的,行为也是完全一致。
- nullptr_t类型数据可以隐式转换成任意一个指针类型。
- nullptr_t类型数据不能转换为非指针类型,即使使用reinterpret_cast<>()的方式。
- nullptr_t类型数据不适用于算数运算表达式。
- nullptr_t类型数据可以用于关系运算表达式,但仅能与nullptr_t类型数据或者指针类型数据进行比较。当且仅当关系运算符为==、>=、<=等时返回true。
总结
在C++中nullptr到任何指针的转换都是隐式的,而(void)0则必须经过类型转换后才能使用。而且nullptr是一个编译期的常量,只是一个关键字,能够为编译器所识别。而(void)0只是一个强制类型转换,返回是一个void*指针类型。
以上是关于C++11之空指针-nullptr的主要内容,如果未能解决你的问题,请参考以下文章
C++11新特性:24—— C++11 nullptr:初始化空指针
C++11新特性nullptr与std::nullptr_t
喵呜:C++基础系列:auto关键字(C++11)基于范围的for循环(C++11)指针空值nullptr(C++11)