0

0

c++中std::call_once的使用场景_c++线程安全的单次初始化机制讲解

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-11-11 13:01:02

|

959人浏览过

|

来源于php中文网

原创

std::call_once 是 C++11 提供的线程安全一次性执行机制,配合 std::once_flag 可确保指定操作仅执行一次。其用于简化单例模式、全局资源初始化等场景,避免竞态条件与重复加锁,支持异常安全:若初始化函数抛出异常,后续调用可重试直至成功;一旦成功,其余调用均直接跳过。该机制取代了易错的双检锁和 volatile 标记方案,使代码更简洁可靠。

c++中std::call_once的使用场景_c++线程安全的单次初始化机制讲解

在C++多线程编程中,某些资源或操作只需要初始化一次,但可能被多个线程同时尝试执行。为了确保这类初始化操作只运行一次且线程安全,std::call_once 提供了一种简洁可靠的机制。

什么是 std::call_once?

std::call_once 是 C++11 引入的工具,定义在 头文件中。它与 std::once_flag 配合使用,保证某个可调用对象(如函数、lambda)在整个程序生命周期中仅被执行一次,即使被多个线程并发调用。

其函数原型为:

template
void call_once(std::once_flag& flag, Callable&& f, Args&&... args);

只要传入同一个 std::once_flag 实例,无论多少线程调用,f 都只会执行一次。

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

典型使用场景:延迟初始化单例

最常见的用途是实现线程安全的单例模式。传统双检锁(Double-Checked Locking)容易出错,而 std::call_once 能简化代码并避免竞态条件。

示例:

class Singleton {
public:
    static Singleton& getInstance() {
        std::call_once(initFlag, [&]() {
            instance.reset(new Singleton);
        });
        return *instance;
    }

private:
    Singleton() = default;
    static std::once_flag initFlag;
    static std::unique_ptr instance;
};

std::once_flag Singleton::initFlag;
std::unique_ptr Singleton::instance;

多个线程同时调用 getInstance() 时,构造函数仅执行一次,无需手动加锁判断。

文心快码
文心快码

文心快码(Comate)是百度推出的一款AI辅助编程工具

下载

全局资源的一次性配置

有些全局服务(如日志系统、配置加载、信号处理注册)只需初始化一次。使用 std::call_once 可以避免重复初始化和数据竞争。

例如:

std::once_flag logInitFlag;

void initLogging() {
    std::call_once(logInitFlag, [](){
        // 打开日志文件、设置格式等
        std::cout     });
}

各模块均可安全调用 initLogging(),实际初始化只发生一次。

替代 volatile 和手动锁的复杂逻辑

过去开发者可能用 volatile bool 标记 + 互斥锁实现“一次执行”,但容易遗漏内存序或死锁。std::call_once 内部已处理所有同步细节,包括异常安全:如果初始化函数抛出异常,call_once 会认为本次调用失败,允许下一次尝试再次执行(直到成功为止)。

这意味着:

  • 若初始化函数抛异常,其他线程仍会被阻塞直到某次调用成功
  • 一旦成功执行一次,后续所有调用直接返回,不再执行函数

这种行为非常适合容错性要求高的初始化流程。

基本上就这些。std::call_once 看似简单,但在构建健壮的多线程程序时非常实用,能有效消除竞态条件,让一次性初始化变得清晰又安全。

相关专题

更多
c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

48

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

95

2025.10.23

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

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

173

2023.11.23

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

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

92

2025.11.27

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

66

2025.10.23

lambda表达式
lambda表达式

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

202

2023.09.15

python lambda函数
python lambda函数

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

187

2025.11.08

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

456

2024.01.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

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

精品课程

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

共58课时 | 3.1万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

ASP 教程
ASP 教程

共34课时 | 3万人学习

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

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