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 中每个对象存在、拥有常地址、保有其最近一次存储值(除非其值不确定),对于 VLA 还有保有其大小 (C99 起)的程序执行部分,被称作该对象的生存期。
对于声明有自动、静态及线程存储期的对象,生存期等于其存储期(注意非 VLA 和 VLA 自动存储期的区别)。
对于拥有分配存储期的对象,其生存期始于分配函数的返回(包含从 realloc 返回),终于 realloc 或解分配函数的调用。注意因为分配的对象没有声明类型,首次访问该对象所用的左值表达式类型会成为其有效类型。
在生存期外访问对象是未定义行为。
int* foo(void) { int a = 17; // a拥有自动存储期 return &a; } // a的生存期结束 int main(void) { int* p = foo(); // p 指向生存期结束后的对象(“悬垂指针”) int n = *p; // 未定义行为 }
指向生存期结束的对象(或该对象后一位置)的指针拥有不确定值。
临时生存期
非左值表达式所指代的拥有数组成员的结构体和联合体对象(直接为其成员或为嵌套的结构体/联合体成员)拥有临时生存期。临时生存期始于求值指代该对象的表达式,终于下一个序列点 (C11 前)包含它的完整表达式或完整声明器结束 (C11 起)。
任何修改临时生存期对象的尝试会导致未定义行为。
struct T { double a[4]; }; struct T f(void) { return (struct T){3.15}; } double g1(double* x) { return *x; } void g2(double* x) { *x = 1.0; } int main(void) { double d = g1(f().a); // C99 : UB 访问生存期结束于 g1 开序列点的 a[0] // C11 : OK , d 为 3.15 g2(f().a); // C99 : UB 修改生存期结束于序列点的 a[0] // C11 : UB 试图修改临时对象 }
引用
- C11 standard (ISO/IEC 9899:2011):
- 6.2.4 Storage durations of objects (p: 38-39)
- C99 standard (ISO/IEC 9899:1999):
- 6.2.4 Storage durations of objects (p: 32)
- C89/C90 standard (ISO/IEC 9899:1990):
- 3.1.2.4 Storage durations of objects