现在c++ 11有了许多新特性。一个有趣而令人困惑的(至少对我来说)是新的nullptr。
不需要讨厌的宏NULL了。
int* x = nullptr;
myclass* obj = nullptr;
不过,我还是不明白nullptr是如何工作的。例如,维基百科的一篇文章说:
c++ 11通过引入一个新的关键字作为区分空指针常量nullptr来纠正这一点。它的类型为nullptr_t,可隐式转换,可与任何指针类型或指针到成员类型相比较。它不能隐式转换,也不能与整型相比,bool类型除外。
它如何既是关键字又是类型的实例?
此外,你是否有另一个例子(除了维基百科的一个),其中nullptr优于好旧的0?
根据cppreference, nullptr是一个关键字,它:
表示指针字面量。它是std::nullptr_t类型的prvalue。
存在从nullptr到空指针值的隐式转换
任何指针类型和任何指向成员类型的指针。类似的转换
存在于任何空指针常量,其中包括类型的值
std::nullptr_t以及宏NULL。
因此nullptr是一个不同类型的值std::nullptr_t,而不是int。它隐式转换为任何指针类型的空指针值。这个神奇的事情发生在您的引擎盖下,您不必担心它的实现。然而,NULL是一个宏,它是一个实现定义的空指针常量。它通常是这样定义的:
#define NULL 0
也就是一个整数。
这是一个微妙但重要的区别,可以避免歧义。
例如:
int i = NULL; //OK
int i = nullptr; //error
int* p = NULL; //OK
int* p = nullptr; //OK
当你有两个像这样的函数重载:
void func(int x); //1)
void func(int* x); //2)
func(NULL)调用1),因为NULL是一个整数。
Func (nullptr)调用2),因为nullptr隐式转换为int*类型的指针。
另外,如果你看到这样的语句:
auto result = findRecord( /* arguments */ );
if (result == nullptr)
{
...
}
你不容易找到findRecord返回什么,你可以确定result必须是一个指针类型;Nullptr使其更具可读性。
在一个推断的背景下,事情的运作有点不同。如果你有一个这样的模板函数:
template<typename T>
void func(T *ptr)
{
...
}
你试着用nullptr调用它:
func(nullptr);
您将得到一个编译器错误,因为nullptr类型为nullptr_t。你必须显式地将nullptr转换为特定的指针类型,或者使用nullptr_t为func提供重载/专门化。
Advantages of using nulptr:
避免函数重载之间的模糊
使您能够进行模板专门化
更安全,直观和富有表现力的代码,例如if (ptr == nullptr)而不是if (ptr == 0)
NULL need not to be 0. As long you use always NULL and never 0, NULL can be any value. Asuming you programme a von Neuman Microcontroller with flat memory, that has its interrupt vektors at 0. If NULL is 0 and something writes at a NULL Pointer the Microcontroller crashes. If NULL is lets say 1024 and at 1024 there is a reserved variable, the write won't crash it, and you can detect NULL Pointer assignments from inside the programme. This is Pointless on PCs, but for space probes, military or medical equipment it is important not to crash.
当一个函数可以接收指向多个类型的指针时,用NULL来调用它是不明确的。这种工作方式现在是非常hack的接受一个int并假设它是NULL。
template <class T>
class ptr {
T* p_;
public:
ptr(T* p) : p_(p) {}
template <class U>
ptr(U* u) : p_(dynamic_cast<T*>(u)) { }
// Without this ptr<T> p(NULL) would be ambiguous
ptr(int null) : p_(NULL) { assert(null == NULL); }
};
在c++ 11中,你可以重载nullptr_t,这样ptr<T> p(42);将是编译时错误,而不是运行时断言。
ptr(std::nullptr_t) : p_(nullptr) { }