标识符
标识符是一个由数字,下划线,大小写拉丁字母和大多数 Unicode 字符组成的任意长度的序列。
有效的标识符的首个字符必须是以下之一:
- 大写拉丁字母 A-Z
- 小写拉丁字母 a-z
- 下划线
- 具有 Unicode 属性 XID_Start 的 Unicode 字符
有效的标识符的其他字符必须是以下之一:
- 数字 0-9
- 大写拉丁字母 A-Z
- 小写拉丁字母 a-z
- 下划线
- 具有 Unicode 属性 XID_Continue 的 Unicode 字符
具有 XID_Start 或 XID_Continue 属性的字符在 DerivedCoreProperties.txt 列出。
标识符区分大小写(小写和大写字母是不同的),而且每一个字符都是起作用的。每个标识符都必须遵循正规化形式 C。
注意:大多数实现中对 Unicode 标识符的支持是受限的,比如 gcc (10 之前)。
在声明中
可以用标识符来命名对象、引用、函数、枚举项、类型、类成员、命名空间、模板、模板特化、形参包 (C++11 起)、goto 标号,以及其他实体,但有以下例外:
- 关键词标识符不能用于其他目的。
|
(C++11 起) |
- 作为特定运算符与标点的代用表示不能用于其他目的。
|
(C++11 起) |
- 有以下形式之一的,作为记号或预处理记号(即不在形如 operator ""id 的用户定义字符串字面量 中)出现的 (C++11 起)标识符被保留:
- 在全局命名空间中,以一个下划线开始的标识符
- 除以下标识符外任何位置带有双下划线或以单下划线后随一个大写字母开始的标识符:
(C++11 起) |
|
(C++11 起) |
(C++20 起) |
这里“被保留”的意思是,标准库的头文件可能 #define 或者声明这样的标识符以便它在内部使用,编译器可能会预先定义这种非标准的标识符,而且名字重整算法可能会假定某些这样的标识符是没有被使用的。如果程序员使用了这样的标识符的话,程序非良构,不要求诊断。
此外,在一个翻译单元 #define 或 #undef 特定名字是未定义行为,详情请见保留宏名。
僵尸标识符
C++14 起,某些标识符被从 C++ 标准库移除。它们列于僵尸名列表。
然而,这些标识符仍然在某些语境中为以前的标准化保留。可移植的代码中,被移除的成员函数名不可用做函数式宏的名字,而其他被移除的成员名不可用作对象式宏的名字。
在表达式中
指名某个变量、函数、概念的特化 (C++20 起)或枚举项的标识符可以作为表达式使用。仅由这个标识符组成的表达式的结果,是该标识符所命名的实体。如果该标识符命名的是某个函数、变量、模板形参对象 (C++20 起)或数据成员,那么表达式的值类别是左值,否则是右值 (C++11 前)纯右值 (C++11 起)(例如枚举项是纯右值表达式,概念的特化是 bool 纯右值 (C++20 起))。
类型
标识表达式的类型就是它指名的实体的类型。
存在以下例外:
void f() { float x, &r = x; [=] { decltype(x) y1; // y1 拥有 float 类型 decltype((x)) y2 = y1; // y2 拥有 float const& 类型 // 因为此 lambda 非 mutable 而 x 是左值 decltype(r) r1 = y1; // r1 拥有 float& 类型 decltype((r)) r2 = y2; // r2 拥有 float const& 类型 }; }
|
(C++11 起) |
无限定的标识符
除了适当声明了的标识符之外,以下各项也可以以相同方式用在表达式中:
- 函数写法的重载运算符名,比如 operator+ 或 operator new;
- 用户定义转换函数的名字,比如 operator bool;
|
(C++11 起) |
- 模板的名字后随它的实参列表,比如 MyTemplate<int>;
- ~ 字符后随类名,比如 ~MyClass;
|
(C++11 起) |
|
(C++26 起) |
这些和标识符一起,被称作无限定的标识表达式。
有限定的标识符
有限定的标识表达式是在无限定的标识表达式前面带上作用域解析运算符 ::,以及以作用域解析运算符分隔的包含以下任意内容的序列(可以为空):
- 命名空间的名字
- 类的名字
|
(C++11 起) |
|
(C++26 起) |
例如,表达式 std::string::npos 是指名命名空间 std 内的类 string 中的静态成员 npos 的表达式。表达式 ::tolower 指名的是全局命名空间内的函数 tolower。表达式 ::std::cout 指名 std 命名空间(顶层命名空间)中的全局变量 cout。表达式 boost::signals2::connection 指名的是在 signals2 命名空间中声明的类型 connection,前者则在命名空间 boost 声明。
有限定标识符中,可能会需要以关键词 template 来消除待决模板名的歧义。
关于为有限定的标识符进行的名字查找的细节,请参见有限定的名字查找。
隐式成员访问转换
如果标识表达式 E 指名了类 C
的某个非静态非类型成员,那么 E 在满足以下所有条件时会转换成类成员访问表达式 this->E:
- E 潜在求值。
- 在 E 处的最内层的外围类是
C
。 - 在 E 处的最内层的外围类从
C
派生。
在模板定义语境(见待决名)中不会应用该转换。
struct X { int x; }; struct B { int b; }; struct D : B { X d; void func() { d; // OK,会转换成 this->d b; // OK,会转换成 this->b x; // 错误:this->x 非良构 d.x; // OK,会转换成 this->d.x // 而不是 d.this->x 或 this->d.this->x } };
名字
名字 是以下各项之一,用来代表某个实体:
- 标识符
- 函数写法的重载运算符的名字(operator+、operator new)
- 用户定义的转换函数的名字(operator bool)
|
(C++11 起) |
- 模板的名字后随它的实参列表(MyTemplate<int>)
每个名字都是由声明引入到程序中来的。在多个翻译单元中使用的相同名字,可以根据它的链接代表相同或者不同的实体。
每当编译器在程序中遇到一个未知的名字时,它就会通过进行名字查找来将它与引入这个名字的声明联系起来,但对模板的声明和定义中的待决名不会这样做。(对于这些名字,编译器需要确定它们命名的是类型、模板还是某些其他实体,这可能需要显式消歧义)。
缺陷报告
下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。
缺陷报告 | 应用于 | 出版时的行为 | 正确行为 |
---|---|---|---|
CWG 1440 | C++11 | 在 :: 之前的 decltype 表达式可以表示任意类型
|
只能表示类和枚举类型 |
CWG 1963 | C++11 | 标识符中可以使用数字,非数字字符和通用字符名以外的由实现定义的字符 | 不能使用 |
CWG 2521 | C++11 | 字面量运算符的用户定义字符串字面量 中的标识符照常保留 | 规则不同 |
CWG 2771 | C++98 | &a 在类语境中不会转换成 &this->a | 会转换 |
CWG 2777 | C++20 | 命名了模板形参对象的标识表达式的类型不明确 | 使之明确 |
CWG 2818 | C++98 | 预定义宏名会被保留 | 不会被保留 |