跟我学C++中级篇——constinit避免SIOF

一、初始化的顺序

Static Initialization Order Fiasco, SIOF。静态初始化顺序问题。这里需要说明的是,静态初始化既包括静态变量也包括全局变量,因为它们都具有静态存储期。在前面文章中分析过,不同的库如果对全局变量有依赖的话,在某些情况下有可能会因初始化的依赖顺序导致意外的结果甚至崩溃。这也就是静态初始化问题的一个具体的表现。

SIOF问题产生的主要原因在于跨单元编译时,互相依赖的静态或全局变量,编译器无法保证其初始化的顺序,这是C++标准本身的原因,所以这个问题一直存在。

比如下面的代码:

c 复制代码
// a.cpp
extern int getData();
int g_A = getData(); // 依赖 b.ccp文件中g_B

// b.cpp
int g_B = 100;
int getData() { return g_B; }

在上述的代码中,如果初始化顺序不正确的话,就会让g_A产生不可控的值。

二、初始化的方式

具有静态存储期的变量在初始化的过程中,主要分为两个阶段:

  1. 静态初始化
    它主要包括零初始化(总提到的全局变量默认为0)和常量初始化。它在编译期完成即保证在程序运行前完成,从而保证了依赖顺序的安全性
  2. 动态初始化
    某些全局变量可能需要构造函数或函数调用的返回值来赋值,这就需要程序运行时才能解决。这时就会产生动态初始化,有可能导致SIOF。

三、constinit避免初始化依赖的原理

constinit要求必须使用常量表达式进行初始化并且在编译完成。constinit的引入,就是让全局变量在编译期的静态初始化这个阶段完成,去除静态初始化顺序中依赖的不确定性。但需要注意的是,其不能完全避免SIOF,除非依赖中的所有变量全部用由constinit限制实现。

四、限制场景

constinit虽然看上去可以解决SIOF,但它也有局限性。对于依赖运行时的值进行动态初始化的相关值时,其无法避免SIOF。同样,如果对于非字面类型,特别是构造函数非constexpr情况的,其同样也无法避免SIOF。

五、总结

如果说灵活是C++的特点,constinit则体现了C++的另外一面。它就是刻板的要求修饰的变量必须在编译期初始化完成。这种刻板恰恰与灵活相辅相成,正如事物的两面性,刻板保证了灵活的安全,灵活动态的使用了刻板。互相成就,千变万化。

相关推荐
rainbow68897 小时前
EffectiveC++入门:四大习惯提升代码质量
c++
秋邱7 小时前
用 Python 写出 C++ 的性能?用CANN中PyPTO 算子开发硬核上手指南
开发语言·c++·python
我在人间贩卖青春8 小时前
C++之析构函数
c++·析构函数
我在人间贩卖青春8 小时前
C++之数据类型的扩展
c++·字符串·数据类型
wangjialelele8 小时前
平衡二叉搜索树:AVL树和红黑树
java·c语言·开发语言·数据结构·c++·算法·深度优先
苏宸啊8 小时前
C++栈和队列
c++
森G9 小时前
七、04ledc-sdk--------makefile有变化
linux·c语言·arm开发·c++·ubuntu
橘颂TA9 小时前
【测试】高效浏览器操作:基础功能与优化设置大全
c++·功能测试·职场和发展·测试·web测试
一只小小的芙厨9 小时前
寒假集训笔记·以点为对象的树形DP
c++·算法
艾莉丝努力练剑10 小时前
hixl vs NCCL:昇腾生态通信库的独特优势分析
运维·c++·人工智能·cann