C 参考手册
- C 语言
- C 的历史
- 基本概念
- 表达式
- 声明
- 初始化
- 函数
- 语句
- 静态断言
- 字符常量
- 函数声明
- 函数定义
- 转义序列
- 翻译阶段
- 标识符
- 作用域
- 生存期
- 查找与命名空间
- ASCII 码表
- 类型
- 遵从性
- 算术类型
- restrict 类型限定符
- 类型
- 对象与对齐
- 主函数
- 未定义行为
- 内存模型
- if 语句
- switch 语句
- for 循环
- while 循环
- do-while 循环
- continue语句
- break 语句
- goto语句
- return 语句
- 值类别
- 求值顺序
- 整数常量
- 浮点常量
- 字符串字面量
- 复合字面量
- 常量表达式
- 隐式转换
- 成员访问运算符
- 逻辑运算符
- 比较运算符
- 算术运算符
- 赋值运算符
- 自增/自减运算符
- 其他运算符
- sizeof 运算符
- _Alignof 运算符
- 转型运算符
- C 运算符优先级
- 泛型选择
- 标量初始化
- 数组初始化
- 结构体与联合体初始化
- 指针声明
- 数组声明
- 枚举
- 存储类指定符
- const 类型限定符
- volatile 类型限定符
- 结构体声明
- 联合体声明
- 位域
- _Alignas
- typedef 声明
- 原子类型
- 外部及试探性定义
- inline 函数指定符
- _Noreturn 函数指定符
- 变长参数
- 内联汇编
- 可分析性
- 替用运算符及记号
- C 关键词
- 预处理器
- C 标准库头文件
- 类型支持
- 程序支持工具
- 变参数函数
- 错误处理
- 动态内存管理
- 日期和时间工具
- 字符串库
- 算法
- 数值
- 文件输入/输出
- 本地化支持
- 原子操作库
- 线程支持库
- 实验性 C 标准库
- 有用的资源
- 符号索引
- 注释
查找与命名空间
在 C 程序中遇到标识符时,会查找定位引入该标识符,并且当前在作用域内的声明。若同一标识符的多个声明属于称作命名空间的相异类别,则 C 允许它们同时在作用域内:
1) 标号命名空间:所有声明为标号的标识符。
4) 所有其他标识符,称之为通常标识符以别于 (1-3) (函数名、对象名、 typedef 名、枚举常量)。
在查找点,根据使用方式确定标识符所属的命名空间:
1) 作为 goto 语句运算数出现的标识符,会在标号命名空间中查找。
2) 跟随关键词 struct 、 union 或 enum 的标识符,会在标签命名空间中查找。
3) 跟随成员访问或通过指针的成员访问运算符的标识符,会在类型成员命名空间中查找,该类型由成员访问运算符左运算数确定。
4) 所有其他标识符,会在通常命名空间中查找。
注意
宏名不是任何命名空间的一部分,因为它们会在语义分析前,为预处理器所替换。
一个常见举措是将 struct/union/enum 名称注入通常命名空间,以 typedef 声明:
struct A { }; // 于标签命名空间中引入名称A typedef struct A A; // 首先,对"struct"后A的查找找到标签命名空间的一个 // 然后将名称A引入通常命名空间 struct A* p; // OK,此A于标签命名空间中查找 A* q; // OK,此A于通常命名空间中查找
众所周知的一个同一标识符横跨两个命名空间使用的示例,是 POSIX 头文件 sys/stat.h
中的标识符 stat 。它用作通常标识符时指名一个函数,而在用作标签时指代一个结构体。
不同于 C++ 中,枚举常量不是结构体成员,而且其命名空间是通常标识符的命名空间,故而 C 中无结构体作用域,其作用域是出现结构体声明的作用域:
struct tagged_union { enum {INT, FLOAT, STRING} type; int integer; float floating_point; char *string; } tu; tu.type = INT; // C 中 OK , C++ 中错误
示例
运行此代码
void foo (void) { return; } // 通常命名空间,文件作用域 struct foo { // 标签命名空间,文件作用域 int foo; // 此 struct foo 的成员命名空间,文件作用域 enum bar { // 标签命名空间,文件作用域 RED // 通常命名空间,文件作用域 } bar; // 此 struct foo 的成员命名空间,文件作用域 struct foo* p; // OK :使用标签/文件作用域名称 "foo" }; enum bar x; // OK :使用标签/文件作用域 "bar" // int foo; // 错误:通常命名空间已有 foo //union foo { int a, b; }; // 错误:标签命名空间已有 foo 在作用域中 int main(void) { goto foo; // OK 从标号命名空间/函数作用域使用 "foo" struct foo { // 标签命名空间,块作用域(于文件作用域中隐藏) enum bar x; // OK ,从标签命名空间/文件作用域使用 "bar" }; typedef struct foo foo; // OK :从标签命名空间/块作用域使用 foo // 定义块作用域通常命名空间的 foo (隐藏于文件作用域) (foo){.x=RED}; // 使用通常命名空间/块作用域 foo 和通常命名空间/文件作用域 RED foo:; // 标号命名空间,函数作用域 }
引用
- C11 standard (ISO/IEC 9899:2011):
- 6.2.3 Name spaces of identifiers (p: 37)
- C99 standard (ISO/IEC 9899:1999):
- 6.2.3 Name spaces of identifiers (p: 31)
- C89/C90 standard (ISO/IEC 9899:1990):
- 3.1.2.3 Name spaces of identifiers