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 标准库
- 有用的资源
- 符号索引
- 注释
泛型选择
提供一种基于控制表达式的类型,在编译时选择数个表达式之一的方法。
语法
_Generic ( controlling-expression , association-list )
|
(C11 起) | ||||||||
其中 association-list 是逗号分隔的关联列表,每个关联拥有语法:
type-name : expression
|
|||||||||
default : expression
|
|||||||||
其中
type-name | - | 任何完整的非可变更改的对象类型(即既非 VLA 亦非指向 VLA 的指针)。 |
controlling-expression | - | 任何表达式(除了逗号运算符),若不使用 default 关联,则其类型必须与 type-name 之一兼容,
|
expression | - | 任何类型和值类别的表达式(除了逗号运算符) |
association-list 中的任意二个 type-name 不能指定兼容类型。使用关键词 default
的关联只能有一个。若不使用 default
,且无一 type-name 与控制表达式类型兼容,则程序无法编译。
解释
首先, controlling-expression 的类型经历左值转换。只在类型域中进行转换:它舍弃顶层 cvr 限定符和原子属性,并应用数组到指针/函数到指针变换到控制表达式的类型,而不实例化任何副效应或计算任何值。
将转换后的类型与来自关联列表中的 type-name 比较。
若其类型与关联之一的 type-name 兼容,则泛型选择的类型、值及值类别就是出现于该 type-name 冒号后的表达式的类型、值及值类别。
若无一 type-name 的类型与 controlling-expression 兼容,且提供了 default
关联,则泛型的类型、值及值类别就是出现于 default :
标号后的表达式的类型、值及值类别。
注意
决不求值 controlling-expression 和未被选择的选项中的 expression 。
因为左值转换, "abc" 匹配 char* 而非 char[4] , (int const){0} 匹配 int 而非 const int 。这是由 C17 DR481 指定的(回溯应用到 C11 )。
泛型选择中的 expression 允许任何值类别,含函数指代器及 void 表达式,而且若它受到选择,则泛型选择自身拥有相同的值类别。
C99 中引入的来自 <tgmath.h>
的泛型数学宏,曾是通过编译器限定的行为实现的。 C11 所引入的泛型选择,给予程序员写相似的类型依赖代码的能力。
泛型选择类似 C++ 中的重载(在编译时基于参数类型选择数个函数之一)。除了它在任意表达式之间选择。
关键词
示例
#include <stdio.h> #include <math.h> // tgmath.h 宏 cbrt 的可能实现 #define cbrt(X) _Generic((X), \ long double: cbrtl, \ default: cbrt, \ float: cbrtf \ )(X) int main(void) { double x = 8.0; const float y = 3.375; printf("cbrt(8.0) = %f\n", cbrt(x)); // 选择默认的 cbrt printf("cbrtf(3.375) = %f\n", cbrt(y)); // 将 const float 转换成 float, // 然后选择 cbrtf }
输出:
cbrt(8.0) = 2.000000 cbrtf(3.375) = 1.500000
引用
- C11 standard (ISO/IEC 9899:2011):
- 6.5.1.1 Generic selection (p: 78-79)