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 标准库
- 有用的资源
- 符号索引
- 注释
隐式转换
当表达式用在期待相异类型的语境中时,可以发生转换:
int n = 1L; // 表达式 1L 拥有类型 long ,期待 int n = 2.1; // 表达式 2.1 拥有类型 double ,期待 int char *p = malloc(10); // 表达式 malloc(10) 拥有类型 void* ,期待 char*
转换在下列情况下发生:
如同赋值的转换
- 在赋值运算符中,右操作数的值被转换成左操作数的无限定类型
- 在标量初始化中,初始化器表达式的值被转换成待初始化对象的无限定类型
- 在对有原型的函数调用表达式中,每个参数表达式的类型被转换成对应参数的声明类型
- 在 return 语句中,
return
操作数的值被转换成拥有函数返回类型的对象
注意在实际赋值中,在转换外,还会移除浮点类型的额外范围和精度,并禁止重叠;这些特性不作用于如同赋值的转换。
默认参数提升
在函数调用表达式中,当调用下列函数时
每个整数类型的参数都会经历整数提升(见后述),而每个 float 类型参数都隐式转换为 double 类型
int add_nums(int count, ...); int sum = add_nums(2, 'c', true); // add_nums 将以三个 int 调用: (2, 99, 1)
注意 float complex 和 float imaginary 在此语境中不会提升到 double complex 和 double imaginary 。
通常算术转换
下列算术运算符的参数会经历隐式转换,为了含有共用实数类型,这是执行计算所用的类型:
- 首先,两个操作数都会经历整数提升(见后述)。然后
- 若两类型在提升后相同,则该类型即为共用类型
- 否则,若两操作数在提升后有相同的符号性(均为有符号或均为无符号),则拥有较低转换等级(见后述)者会隐式转换成拥有较高转换等级的操作数的类型
- 否则,两者符号性不同:若无符号类型操作数拥有大于或等于有符号类型操作数的转换等级,则有符号类型操作数会隐式转换成无符号类型
- 否则,两者符号性不同且有符号操作数的等级大于无符号操作数的等级。此情况中,若有符号类型可以表达无符号类型的所有值,则有无符号类型的操作数被隐式转换成有符号操作数的类型。
- 否则,两个操作数都会经历隐式转换,到有符号类型的无符号类型对应者。
1.f + 20000001; // int 被转换成 float ,给出 20000000.00 // 相加后舍入到 float ,给出20000000.00 (char)'a' + 1L; // 首先,char 被提升回 int。 // 这是有符号+有符号的情形,等级不同 // int 被转换成 long ,结果是 signed long 的 98 2u - 10; // 有符号/无符号,等级相同 // 10 被转换成无符号,无符号数学运算为模 UINT_MAX+1 // 对于 32 位 int ,结果是 unsigned int 类型的 4294967288 (即 UINT_MAX-7 ) 0UL - 1LL; // 有符号/无符号相异等级,有符号的等级较大。 // 若 sizeof(long) == sizeof(long long) ,则有符号数不能表示所有无符号数 // 这是最后一种情况:两个操作数都被转换成 unsigned long long // 结果是 unsigned long long 类型的 18446744073709551615( ULLONG_MAX )
结果类型按下列方式确定:
- 若两操作数均为复数,则结果类型是复数
- 若两操作数均为虚数,则结果类型是虚数
- 若两操作数均为实数,则结果类型为实数
- 若两浮点操作数拥有不同定义域(复数 VS 实数、复数 VS 虚数,或虚数 VS 实数),则结果类型是复数
double complex z = 1 + 2*I; double f = 3.0; z + f; // z 保持原态,f 被转换成 double ,结果是 double complex
浮点运算符的结果可能照常会有大于其类型所指示的范围和及精度(见 FLT_EVAL_METHOD )。
注意:实数和虚数操作数不会隐式转换成复数,因为这么做需要额外计算,会在牵涉到无穷大、NaN和有符号零的具体情况时产生不想要的结果。例如,若实数被转换成复数,2.0×(3.0+i∞)会按照(2.0+i0.0)×(3.0+i∞) ⇒ (2.0×3.0–0.0×∞) + i(2.0×∞+0.0×3.0) ⇒ NaN+i∞求值,而非正确的6.0+i∞。若虚数被转换成复数,则i2.0×(∞+i3.0)会按照(0.0+i2.0) × (∞+i3.0) ⇒ (0.0×∞ – 2.0×3.0) + i(0.0×3.0 + 2.0×∞) ⇒ NaN + i∞求值,而非–6.0 + i∞。
注意:无关乎通常算术转换,可以在如同规则下,始终以窄于这些规则指定的类型进行计算。
值变换
左值转换
任何非数组类型的左值表达式,在用于异于下列语境时
会经历左值转换:类型保持相同,但失去 const/volatile/restrict 限定符及原子属性,若原先有。值保持相同,但失去其左值属性(不再能取其地址)。
若左值拥有不完整类型,则行为未定义。
若左值指代自动存储期的对象,该对象从不被取地址,且若该对象未被初始化(没有用初始化器声明且没有在使用它前赋值),则行为未定义。
此转换模拟从内存中其位置加载对象的值。
volatile int n = 1; int x = n; // n 上左值转换读 n 的的值 volatile int* p = &n; // 无左值转换:不读 n 的值
数组到指针转换
会经历到指向其首元素的非左值指针的转换。
若数组声明为 register ,则行为未定义。
int a[3], b[3][4]; int* p = a; // 转换成 &a[0] int (*q)[4] = b; // 转换成 &b[0]
函数到指针转换
任何函数指代器表达式,在用于异于下列语境时
会经历到指向表达式所指代函数的指针的转换。
int f(int); int (*p)(int) = f; // 转换成 &f (***p)(1); // 重复解引用到 f 和转换回 &f
隐式转换语义
隐式转换,要么是如同赋值要么是通常算术转换,由二阶段组成:
兼容类型
将任何类型的值转换成任何兼容类型始终是无操作,且不改变表示。
uint8_t (*a)[10]; // 若 uint8_t 是对 unsigned char 的 typedef unsigned char (*b)[] = a; // 则这些指针类型是兼容的
整数提升
整数提升是任何等级小于或等于 int 等级的整数类型,或是 _Bool 、 signed int 、 unsigned int 类型的位域类型的值到 int 或 unsigned int 类型值的隐式转换。
若 int 能表示原类型的整个值域(或原位域的值域),则值转换成 int 类型。否则值转化成 unsigned int 类型。
整数提升保持值,包含符号:
int main(void) { void f(); // 旧式函数声明 char x = 'a'; // 整数转换 int 到 char f(x); // 整数提升 char 回 int } void f(x) int x; {} // 函数期待int
上述的等级是每个整数类型的属性,定义如下:
注意:整数提升仅应用于
- 通常算术转换的一部分(见前述)
- 默认参数提升的一部分(见前述)
- 给一元算术运算符 + 和 - 的操作数
- 给一元位运算符的操作数
- 给位移运算符 << 和 >> 的两个操作数
布尔转换
任何标量类型的值可以隐式转换成 _Bool 。比较等于零的值转换成 0 ,所有其他值转换成 1
bool b1 = 0.5; // b1 == 1 (0.5 转换成 int 会是零) bool b2 = 2.0*_Imaginary_I; // b2 == 1 (但转换成 int 会是零) bool b3 = 0.0 + 3.0*I; // b3 == 1 (但转换成 int 会是零) bool b4 = 0.0/0.0; // b4 == 1 (NaN 与零比较不相等)
整数转换
任何整数类型的值可以隐式转换到任何其他整数类型。除了上述整数提升和布尔转换所提及的情况,规则为:
- 若目标类型能表示值,则值不变
- 否则,若目标类型为无符号,则源值会重复减或加值 2b
,其中 b 是目标类型的位数,直到结果符合目标类型。换言之,无符号整数实现模算术。 - 否则,若目标类型为有符号,则行为是实现定义的(可能包括引发信号)
char x = 'a'; // int -> char,结果不变 unsigned char n = -123456; // 目标是无符号数,结果为 192 (即 -123456+483*256 ) signed char m = 123456; // 目标是有符号数,结果实现定义 assert(sizeof(int) > -1); // 断言失败: // 运算符 > 要求 -1 到 size_t 的转换, // 目标为无符号,结果是 SIZE_MAX
实浮点整数转换
任何实浮点类型的有限值可以隐式转换到任何整数类型。除了上述布尔转换所提及的情况,规则为:
- 忽略小数部分(向零取整)。
- 若结果值可表示成目标类型,则使用该值
- 否则,行为未定义
int n = 3.14; // n == 3 int x = 1e10; // 对于 32 位 int 是未定义行为
任何整数类型的值可以隐式转换成任何实浮点类型。
- 若值能被目标类型准确表示,则它不变
- 若值能被表示,但无法准确表示,则结果是最接近的较高或较低值(换言之,舍入方向是实现定义的),尽管若支持IEEE算术,则向最近舍入。此情况下是否引发 FE_INEXACT 是未指定的。
- 若值不能被表示,则行为未定义,尽管若支持 IEEE 算术,则引发 FE_INVALID 且值为未指定。
此转换的结果可能拥有大于其目标类型所指示的值和精度(见 FLT_EVAL_METHOD )。
若在浮点到整数转换中控制 FE_INEXACT ,则可以使用 rint 及 nearbyint 。
double d = 10; // d = 10.00 float f = 20000001; // f = 20000000.00 (FE_INEXACT) float x = 1+(long long)FLT_MAX; // 未定义行为
实浮点数转换
任何实浮点类型的值可以隐式转换到任何其他实浮点类型。
- 若值能被目标类型准确表示,则它不变
- 若值能被表示,但无法准确表示,则结果是最接近的较高或较低值(换言之,舍入方向是实现定义的),尽管若支持IEEE算术,则向最近舍入
- 若值不能被表示,则行为未定义
本节未完成 原因:检查 IEEE ,是否适用于有符号无穷大 |
此转换的结果可能拥有大于其目标类型所指示的值和精度(见 FLT_EVAL_METHOD )。
double d = 0.1; // d = 0.1000000000000000055511151231257827021181583404541015625 float f = d; // f = 0.100000001490116119384765625 float x = 2*(double)FLT_MAX; // 未定义
复数类型转换
任何复数类型的值可以隐式转换成任何另一种复数类型。实部和虚部各自遵循实浮点类型的转换规则。
虚数类型转换
任何虚数类型的值可以隐式转换成另一种虚数类型。虚部遵循实浮点类型的转换规则。
double imaginary d = 0.1*_Imaginary_I; float imaginary f = d; // f 为 0.100000001490116119384765625*I
实复数转换
任何实浮点类型的值可以隐式转换成任何复数类型。
- 结果的实部根据实浮点类型的转换规则确定
- 结果的虚部是正零(或非 IEEE 系统上的无符号零)
任何复数类型的值可以隐式转换成任何实浮点类型
- 实部遵循实浮点类型的规则转换
- 虚部被忽略
注意:在复到实转换中,虚部的 NaN 不会传播到实结果。
double complex z = 0.5 + 3*I; float f = z; // 舍去虚部,设置 f 为 0.5 z = f; // 设置 z 为 0.5 + 0*I
实虚数转换
任何虚数类型的值可以转换到任何实数类型(整数或浮点数)。结果始终是正(或无符号)零,除非目标类型是 _Bool ,这种情况下会应用布尔转换规则。
任何实数类型的值可以隐式转换成任何虚数类型。结果始终是虚数正零。
double imaginary z = 3*I; bool b = z; // 布尔转换:设置 b 为 true float f = z; // 实虚转换:设置 f 为 0.0 z = 3.14; // 虚实转换:设置 z 为 0*_Imaginary_I
复虚数转换
任何虚数类型的值可以隐式转换到任何复数类型。
- 结果的实部是正零
- 结果的虚部遵循对应的实类型转换规则
任何复数类型可以隐式转换到任何虚数类型
- 实部被忽略
- 结果的虚部遵循对应的实数类型转换规则
double imaginary z = I * (3*I); // 复结果 -3.0+0i 失去实部 // 设置 z 为 0*_Imaginary_I
指针转换
指向 void 可与任何指向对象指针类型间相互隐式转换,并拥有下列语义:
- 若指向对象的指针被转换成指向void的指针再转换回来,则其值与原指针比较相等。
- 不提供其他保证
int* p = malloc(10 * sizeof(int)); // malloc 返回 void*
指向无限定类型的指针可以隐式转换成指向该类型有限定版本的指针(换言之,可以添上 const 、 volatile 、 及 restrict 限定符)。原指针与结果比较相等。
int n; const int* p = &n; // &n 拥有类型 int*
任何拥有值 0 的整数常量表达式也是一个拥有转换成 void* 类型的零的整数指针表达式,可以隐式转换成任意指针类型(既可以是指向对象指针,又可以是指向函数指针)。结果是该类型的空指针值,保证与任何该类型的非空指针值比较不相等。此整数或 void* 表达式又称空指针常量,而且标准库提供此常量作为宏 NULL 的一种定义。
int* p = 0; double* q = NULL;
注意
尽管有符号整数在任何算术运算符中的溢出是未定义行为,在整数类型转换中溢出有符号整数仅是未指定行为。
另一方面,尽管任何算术运算符(和整数转换)中无符号整数溢出是良好定义的操作,并遵循模算术规则,在浮点到整数转换中溢出无符号整数是未定义行为:可以转换成无符号整数的实浮点类型值是来自开区间 (-1; Unnn_MAX+1) 的值。
unsigned int n = -1.0; // 未定义行为
指针和整数间(除了从指针到 _Bool 和从拥有零值的整数常量表达式到指针)、指向对象指针间(除了从或到指向 void 的指针)以及指向函数指针间(除非函数拥有兼容类型)的转换始终非隐式,并要求有转型运算符。
不存在(隐式或显式的)指向函数指针与指向对象指针(包括 void* )或整数间的转换。
引用
- C11 standard (ISO/IEC 9899:2011):
- 6.3 Conversions (p: 50-56)
- C99 standard (ISO/IEC 9899:1999):
- 6.3 Conversions (p: 42-48)
- C89/C90 standard (ISO/IEC 9899:1990):
- 3.2 Conversions