C++ 未定义行为成因列表
cppreference.com: "Undefined behavior renders the entire program meaningless if certain rules of the language are violated."
未定义的行为(Undefined Behavior)指程序不可预测的执行效果,一般由错误的代码实现引起。出于效率、兼容性等多方面原因,语言标准不便于定义错误程序的明确行为,而是将其统称为“未定义”的行为,可以是崩溃,也可以是非预期的任意表现,有些问题在编译器和执行环境的特殊处理下也可能不会造成实质性的危害,但不具备可移植性。代码质量管理的一个重要目标就是消除会导致未定义行为的代码。
C++ 标准声明了导致未定义行为的情况,本文梳理了 ISO/IEC 14882:2003 和 ISO/IEC 14882:2011 前 18 章的相关内容,后 12 章的主题为标准库实现,相关内容的主旨在前 18 章中已有体现。
下列问题会导致未定义的行为:
- 代码行以反斜杠结尾,且与下一行连接后生成通用字符名称
- 非空源文件未以换行符结尾,或以换行符结尾但换行符之前是反斜杠
- 连接预处理符号时产生了通用字符名称
- 存在不符合词法的单引号或双引号
- 在 #include 指令中,'、"、\、//、/* 出现在定界符 < > 之间,或 '、\、//、/* 出现在定界符 " 之间
- 无后缀 10 进制整数字面常量超过 long int 取值范围
- 使用非标准转义字符
- 修改字符串字面常量
- 窄字符串与宽字符串连接
- 违反 One Definition Rule
- 具有静态或线程存储期的对象在析构函数中调用 std::exit 函数
- 函数内具有静态或线程存储期的对象已析构,之后该函数又被调用并引用到已析构的对象
- 在对象析构之后使用对象
- 指针指向长度为 0 的内存空间并被解引用
- 内存回收函数抛出异常
- 使用不匹配的方法分配回收资源
- 使用已被释放的指针
- 对象生命周期已结束,但未调用其有副作用的析构函数
- 在分配空间后,生命周期开始前,或在生命周期结束后,回收空间前,通过指针访问对象
- 在分配空间后,生命周期开始前,或在生命周期结束后,回收空间前,通过 glvalue 访问对象
- 具有静态、线程或自动存储期和 non-trivial 析构函数的对象,其空间被非兼容类型的对象占据
- 常量对象的空间或曾属于常量对象的空间被其他对象占据
- 通过 glvalue 访问对象,但 glvalue 的类型不符合要求
- 通过 glvalue 引用不相关类型的对象或未初始化的对象
- 浮点类型转换产生的结果无法在相应的空间中表示
- 浮点类型转为整数类型时,整数类型无法存储浮点类型的整数部分
- 整数类型转为浮点类型时,浮点类型无法存储整数的值
- 表达式求值依赖无确定顺序的副作用
- 表达式的结果在数学上没有定义
- 被调用函数的语言链接性与该函数定义的语言链接性不符
- 将非 POD 对象传入可变参数列表
- 用 static_cast 将基类引用转为派生类引用,基类为虚基类,或引用的实际对象并非派生类对象
- 用 static_cast 将基类指针转为派生类指针,基类为虚基类,或指向的实际对象并非派生类对象
- 用 static_cast 将成员指针转为基类成员指针时,基类中没有相关成员
- 函数指针被转为不兼容的类型并执行
- 类型转换时去掉对象 const 属性并修改对象
- 对不完整类型的对象取地址,但该对象的完整类型重载了 operator &
- new 运算符第一个数组维度的参数为负数
- 用 delete 释放数组漏写中括号,用 delete 释放对象多写中括号,用 delete 释放非 new 表达式的结果
- 被 delete 释放的对象或数组类型不符合要求
- 用 delete 释放不完整类型的对象,但在对象完整类型声明中有 non-trivial 析构函数
- 对象解引用成员指针时,对象的动态类型不包含成员指针引用的成员
- 对象解引用成员指针时,成员指针为空指针
- / 或 % 运算符第二个操作数的值为 0
- 指针加减整数,结果超出了指针指向数组的地址范围,使指针的值溢出
- 未指向同一数组的指针相减
- 移位运算符右操作数为负数或超过相关类型比特位的数量
- 对有符号整数进行超出取值范围的左移运算
- 将对象的值赋给具有部分重叠区域的另一个对象
- 有返回值的函数没有通过 return 语句返回
- 递归地定义和初始化静态对象
- 修改非 mutable 常量对象
- 使用没有 volatile 限定的 glvalue 访问有 volatile 限定的对象
- 具有 noreturn 属性的函数返回至调用方
- 空指针解引用
- 对象的实际类型与当前静态类型不相关,并调用其非静态成员函数
- 在构造函数或析构函数中调用纯虚函数
- 对象的实际类型与当前静态类型不相关,并调用其析构函数
- 在对象生命周期结束后调用其析构函数
- 基类对象构造完毕之前调用成员函数
- 对成员或基类对象的不合理引用
- 将对象指针转为其基类对象的指针时,基类对象尚未开始构造或已结束析构
- 正在构造或析构的对象通过 . 或 -> 调用虚函数,而且该对象与当前构造或析构函数不属于同一个类或基类
- typeid 作用于正在构造或析构的对象,而且该对象与当前构造或析构函数不属于同一个类或基类
- dynamic_cast 作用于正在构造或析构的对象,而且该对象与当前构造或析构函数不属于同一个类或基类
- 对模板函数进行非良构调用,或在模板定义和实例化上下文之外有更好的候选函数匹配
- 需要无限递归的模版实例化
- 构造或析构函数在 function-try-block 的 handler 中访问非静态成员
- 有返回值的函数在 function-try-block 的 handler 中没有正确返回
- 在 #if、 #elif 的条件中,由宏展开产生了 defined 表达式,或 defined 表达式格式不正确
- #include 指令经宏展开后不满足标准格式
- 宏的实参列表中出现预处理指令
- 预处理运算符 # 的结果不是有效的字符串
- 预处理运算符 ## 的结果不是有效的符号
- #line 指定的行号为 0 或大于规定值
- #line 指令不符合标准格式
- 用 #define 定义或用 #undef 取消定义具有保留意义的名称
- 供语言机制调用的函数不符合要求
- 程序实现了应由标准库提供的功能
- 未经许可,向 std 命名空间添加声明或定义
- 对标准库,特化模板类成员函数,特化模板类成员模板函数,特化、偏特化成员类模版
- 未经许可,向 posix 命名空间添加声明或定义
- 声明或定义标准库保留名称
- 编译器未提供标准头文件,但编译时引入了非标准同名头文件
- 为标准库函数提供不符合要求的参数
- 多线程调用标准库函数造成数据竞争
- 违反标准库函数要求的前置条件,除非标准库函数声明了这种情况会抛出异常
- offsetof 用于非 standard layout 类型,或用于计算静态成员以及成员函数的偏移量
- 可变参数列表中省略号的前一个形式参数为引用、数组、函数,或具有与默认参数提升后不兼容的类型
- longjmp 跳转使应被执行的析构函数未被执行