跟我学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++的另外一面。它就是刻板的要求修饰的变量必须在编译期初始化完成。这种刻板恰恰与灵活相辅相成,正如事物的两面性,刻板保证了灵活的安全,灵活动态的使用了刻板。互相成就,千变万化。

相关推荐
郝学胜-神的一滴5 小时前
Qt 高级开发 009: C++ Lambda 表达式
开发语言·c++·qt·软件构建
石山代码5 小时前
C++ 轻量级日志系统
开发语言·c++
王老师青少年编程8 小时前
2026年全国青少年信息素养大赛初赛真题(算法应用主题赛C++初中组初赛真题3:文末附答案和解析)
c++·真题·答案·初赛·2026年·青少年信息素养大赛·初中组
轻颂呀9 小时前
C++11——并发库介绍
开发语言·c++
梓䈑10 小时前
【算法题攻略】快速排序 和 归并排序
数据结构·c++·排序算法
fan_music11 小时前
设计模式学习
c++·设计模式
小小编程路11 小时前
C++ 常用逻辑运算符
开发语言·c++·算法
‎ദ്ദിᵔ.˛.ᵔ₎13 小时前
C++ 智能指针
开发语言·c++
Lumbrologist13 小时前
【C++】零基础入门 · 第 4 节:循环结构(while、for、do-while)
开发语言·c++
我命由我1234513 小时前
Android Framework P4 - ServiceManager 进程
android·c语言·c++·visualstudio·android studio·android-studio·android runtime