0

0

c++中如何使用std::call_once确保全局唯一初始化_c++单例技巧【实例】

穿越時空

穿越時空

发布时间:2026-01-22 17:52:02

|

973人浏览过

|

来源于php中文网

原创

std::call_once通过std::once_flag的原子状态和平台同步原语实现“首次调用者赢”,仅一个线程执行callable,其余等待;正确使用需满足三要素:once_flag静态存储期、callable不抛异常、所有线程共享同一flag内存。

c++中如何使用std::call_once确保全局唯一初始化_c++单例技巧【实例】

std::call_once 为什么能保证只执行一次

它底层依赖 std::once_flag 的原子状态和平台级的线程同步原语(如 futex 或 Windows SRW lock),不是靠锁住整个函数,而是通过“首次调用者赢”的机制:多个线程同时抵达时,只有一个线程真正执行传入的 callable,其余阻塞等待该 callable 返回后才一起继续——所以初始化逻辑不会重复执行,也不会出现竞态。

正确声明和使用 std::call_once 的三要素

漏掉任意一个都会导致编译失败或未定义行为:

  • std::once_flag 必须是 静态存储期(全局、静态局部、静态成员),不能是变量或堆分配对象
  • 传给 std::call_once 的 callable 必须可调用(函数指针、lambda、functor),且不能抛异常(否则程序 terminate)
  • 必须确保 std::once_flag 在所有线程中访问的是同一块内存(即不能每个线程都 new 一个)
std::once_flag init_flag;
std::string* g_config_ptr = nullptr;

void init_config() { g_config_ptr = new std::string("loaded"); }

// 正确:静态 once_flag + 全局作用域调用 void get_config() { std::call_once(init_flag, init_config); // 此时 g_config_ptr 已安全初始化 }

在单例类中用 std::call_once 实现线程安全 getInstance()

比双重检查锁定(DCLP)更简洁、不易出错,且 C++11 起标准保证其正确性。关键点在于把初始化逻辑完全交给 std::call_once,不手动管理指针或锁。

class ConfigSingleton {
    static std::once_flag m_init_flag;
    static ConfigSingleton* m_instance;
ConfigSingleton() { /* 构造可能耗时或有副作用 */ }

public: static ConfigSingleton& getInstance() { std::call_once(m_init_flag, []() { m_instance = new ConfigSingleton(); }); return *m_instance; } };

// 定义静态成员(必须在 .cpp 中) std::once_flag ConfigSingleton::m_init_flag; ConfigSingleton* ConfigSingleton::m_instance = nullptr;

注意:m_init_flagm_instance 都必须定义在类外;lambda 捕获为空,避免隐式捕获引发生命周期问题;构造函数不应抛异常,否则 std::call_once 会终止程序。

Remove.bg
Remove.bg

AI在线抠图软件,图片去除背景

下载

立即学习C++免费学习笔记(深入)”;

常见误用和崩溃场景

这些错误在多线程下极难复现,但一旦发生就是 crash 或数据错乱:

  • std::once_flag 声明为自动变量:void foo() { std::once_flag flag; std::call_once(flag, ...); } → 每次调用新建 flag,完全失去“once”语义
  • 在不同翻译单元中定义同名静态 std::once_flag → ODR 违反,链接时可能合并也可能不合并,行为未定义
  • callable 中抛出未捕获异常 → std::call_once 直接调用 std::terminate(),进程退出
  • std::call_once 初始化需要析构的对象(如文件句柄、socket)→ 它不提供销毁机制,单例生命周期无法与程序结束对齐

如果单例需要资源清理,要么用静态局部变量(C++11 起线程安全且自动析构),要么额外设计 shutdown 流程,别指望 std::call_once 管释放。

相关专题

更多
javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

97

2025.11.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

205

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

190

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

50

2026.01.05

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

393

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

573

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

393

2023.07.18

Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

9

2026.01.22

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.6万人学习

Excel 教程
Excel 教程

共162课时 | 12.9万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号