引出
c++11提供了nullptr来代替NULL 去表示空指针。
以下讨论为何抛弃。
本文参考了:现代c++教程 作者:欧长坤
原本为什么用NULL
NULL是表示一个空指针,表示这个指针变量不指向任意内存地址,本质是是沿用的c中的做法。
NULL的含义
在c中
在c中,NULL表示 ((void*)0) 或 0,这是由标准库实现的
在c++中
而在c++中,由于不允许 void* 隐式转换到其他指针类型,也就是:
void* a = 0;
char *ch = a;//不允许
a不能直接赋值给ch,因为c++不允许隐式转换指针,必须要强转,或者使用c++11给出的reinterpret_cast。
这是好事,因为c中可以这么做,导致了很多隐藏的bug
既然(void*)0 不能直接赋值,那么如果NULL的值还是(void*)0 ,就不能实现:
char * ch = NULL;
这样的操作,因此,c++中的NULL,只能是0
NULL 为0带来的问题
看这段代码:
void show(char * ch) {
cout << "show1" << endl;
}
void show(int a) {
cout << "show2" << endl;
}
int main() {
char * ch = NULL;
show(ch);
}
ch的值是NULL,也就是0,是int,那么在调用重载函数时,到底调用哪一个呢?
我在vs2017测试时,发现调用正确,是调用的void show(char * ch),但是根据作者的说法,会调用到void show(int a)。那么可能是环境不同导致的。
这就导致了问题:到底调用的是谁,就不明确了
因此引入nullptr去代替NULL。
nullptr 的类型是什么
nullptr 的类型为 nullptr_t,能够隐式的转换为任何指针或成员指针的类型
nullptr解决的问题
1.nullptr能够直接赋值给任意类型的指针,而不需要强转
char* p =nullptr;
2.nullptr能区分出 空指针 和 0
像是下面这段代码,没有歧义
void show(char * ch) {
cout << "show1" << endl;
}
void show(int a) {
cout << "show2" << endl;
}
int main() {
char * ch = nullptr;
show(ch);
show(2);
}
3.能够直接使用nullptr 进行相等和不相等的比较
char * ch = nullptr;
if (ch==nullptr)
{
cout << "ch 是空指针" << endl;
}
else
{
cout << "ch 非空指针" << endl;
}
总结
由于nullptr能够无缝代替NULL,且能规避掉一些含义不清晰问题,因此建议直接把NULL 替换为nullptr