在 C++98 中,空指针由字面量表示0
(或者实际上是任何值为 0 的常量表达式)。在 C++11 中,我们更喜欢nullptr
。但这不适用于纯虚函数:
struct X
{
virtual void foo() = nullptr;
};
为什么这不起作用?这不是完全有道理吗?这仅仅是一个疏忽吗?会修复吗?
在 C++98 中,空指针由字面量表示0
(或者实际上是任何值为 0 的常量表达式)。在 C++11 中,我们更喜欢nullptr
。但这不适用于纯虚函数:
struct X
{
virtual void foo() = nullptr;
};
为什么这不起作用?这不是完全有道理吗?这仅仅是一个疏忽吗?会修复吗?
因为语法说0
,不是表达式或其他一些非终端匹配nullptr
。
因为0
一直都有效。Even0L
格式不正确,因为它与语法不匹配。
编辑
Clang 允许= 0x0
,= 0b0
和= 00
(31.12.2013)。这是不正确的,当然应该在编译器中修复。
函数的= 0
符号virtual
并不是字面上的“赋值 null”,而是一种实际上具有欺骗性的特殊符号:也可以实现纯虚函数。
abstract
使用各种上下文关键字,允许而不是= nullptr
成为上下文关键字会更有意义abstract
。
语法就是这样定义的,如果我们看一下C++标准草案的9.2
Class members,相关语法如下:
[...]
member-declarator:
declarator virt-specifier-seqopt pure-specifieropt
[...]
pure-specifier:
= 0
^^^
语法特别指出纯说明符是= 0
而不是整数文字或表达式,这似乎没有留下任何回旋余地。如果我尝试这样的事情:
virtual void foo() = 0L;
或者:
virtual void foo() = NULL ;
gcc
告诉我:
错误:';'之前的纯说明符无效(只允许'= 0')令牌
并clang
说:
错误:函数的初始化器看起来不像纯说明符
尽管以下内容在两者中都有效:
#define bar 0
//...
virtual void foo() = bar;
似乎也clang
允许八进制文字、十六进制文字和二进制文字零,这是不正确的行为。
更新
显然Visual Studio
接受NULL
和任何零整数文字,包括0L
,等......虽然它不接受.0x0
00
nullptr
= 0
那里有固定的含义。它不是真正的整数零。因此,您不能像那样简单地替换它。
该= 0
语法不用于初始化指针,它只是在语法上表明提供的virtual
是纯的。
因此= 0
,声明 pure virtual
s 的语法没有改变。
的全部nullptr
(或无论如何大部分)是只能分配给(或用于初始化)指针。
在这种情况下,您没有初始化或分配给指针,因此在这种情况下您能够使用它甚至没有意义。
这并不意味着它是一个指针,或者它必须等于nullptr
.
= 0
是足够的并且意味着虚函数是纯的。
C++11 语法只允许0
在这里(并不意味着指针)。nullptr
不是0
,它失败了。NULL
仅在NULL
定义为时才有效0
(有时是这种情况,但并非总是如此)。只需0
在此处使用,或使用以下定义(如果您真的想使用空值,当它不是指针时)。
#define VIRTUAL_NULL 0
struct X
{
virtual void foo() = VIRTUAL_NULL;
};