0

0

c++怎么利用std::call_once确保初始化一次_c++ 多线程环境单例安全加载【方法】

冰火之心

冰火之心

发布时间:2026-01-09 12:22:03

|

372人浏览过

|

来源于php中文网

原创

std::call_once能保证只执行一次,因其内部采用原子操作加互斥锁双重机制,确保多线程下仅一个线程执行可调用对象,其余阻塞等待;正确使用需满足三条件:once_flag须为静态存储期、可调用对象不可抛异常、多线程共享同一flag实例。

c++怎么利用std::call_once确保初始化一次_c++ 多线程环境单例安全加载【方法】

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

因为 std::call_once 内部用原子操作 + 互斥锁双重机制检测状态:只要某个 std::once_flag 对象被传入并配合一个可调用对象,无论多少线程并发调用,最终只有**一个线程**真正执行该可调用体,其余全部阻塞等待,直到初始化完成才继续。它不依赖用户手动加锁,也无需判断“是否已初始化”,语义更干净。

正确使用 std::call_once 的三个必要条件

缺一不可,否则可能崩溃、重复执行或死锁:

  • std::once_flag 对象必须是 静态存储期(全局、静态局部、类静态成员),不能是上临时变量或每次调用都新建的
  • 传给 std::call_once 的可调用对象(如 lambda、函数指针)不能抛异常;若抛了,该 std::once_flag 永远处于“未就绪”状态,后续所有调用都会直接抛 std::system_error(错误码为 std::errc::invalid_argument
  • 多个线程必须共享同一个 std::once_flag 实例,不能各自持有一份副本

单例构造中 std::call_once 的典型写法

常见于延迟初始化的线程安全单例。注意静态局部变量本身已有线程安全保证(C++11 起),但 std::call_once 更适合需要控制初始化时机、或初始化逻辑跨多个步骤的场景:

class Singleton {
public:
    static Singleton& instance() {
        std::call_once(init_flag_, []() {
            instance_ = new Singleton();
        });
        return *instance_;
    }

private: Singleton() = default; static Singleton* instance_; static std::once_flag initflag; };

Singleton* Singleton::instance_ = nullptr; std::once_flag Singleton::initflag;

这里 instance_ 是裸指针,实际项目中建议用 std::unique_ptr 管理;init_flag_ 必须定义在类外,否则链接失败。

TemPolor
TemPolor

AI音乐生成器,一键创作免版税音乐

下载

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

std::call_once 和 static local variable 初始化的区别

两者都能实现线程安全的首次调用初始化,但行为不同:

  • static Singleton& instance() { static Singleton inst; return inst; }:初始化发生在第一次进入函数时,且由编译器生成 guard 变量保障,无需手动管理 flag;但无法捕获初始化失败、不能做多步协调(比如先建配置再建实例)
  • std::call_once:初始化时机完全可控,可放在任意位置(比如构造函数里、某次网络响应后);支持多个初始化动作共用一个 flag;但需自己确保 flag 生命周期和异常安全
  • 性能上,static local 首次调用略慢(多一次 guard 检查),之后无开销;std::call_once 每次调用都有原子读+分支判断,但现代实现优化后差距极小

真正容易被忽略的是:如果初始化函数里调用了另一个也依赖 std::call_once 的模块,而两个 flag 初始化顺序没理清,可能引发静态初始化顺序 fiasco —— 这种问题不会报错,但行为未定义。

相关专题

更多
lambda表达式
lambda表达式

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

202

2023.09.15

python lambda函数
python lambda函数

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

189

2025.11.08

Python lambda详解
Python lambda详解

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

40

2026.01.05

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

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

382

2023.07.18

堆和栈区别
堆和栈区别

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

567

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

478

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

Golang 分布式缓存与高可用架构
Golang 分布式缓存与高可用架构

本专题系统讲解 Golang 在分布式缓存与高可用系统中的应用,涵盖缓存设计原理、Redis/Etcd集成、数据一致性与过期策略、分布式锁、缓存穿透/雪崩/击穿解决方案,以及高可用架构设计。通过实战案例,帮助开发者掌握 如何使用 Go 构建稳定、高性能的分布式缓存系统,提升大型系统的响应速度与可靠性。

60

2026.01.09

java学习网站推荐汇总
java学习网站推荐汇总

本专题整合了java学习网站相关内容,阅读专题下面的文章了解更多详细内容。

61

2026.01.08

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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