0

0

c++的std::call_once和std::once_flag怎么用 保证函数只执行一次【多线程】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-08 16:19:02

|

575人浏览过

|

来源于php中文网

原创

std::call_once和std::once_flag确保某函数在多线程下仅执行一次,首次调用时执行、后续直接返回;once_flag须为静态或全局变量,call_once支持异常安全和参数转发,但不替代互斥锁。

c++的std::call_once和std::once_flag怎么用 保证函数只执行一次【多线程】

std::call_oncestd::once_flag 是 C++11 引入的线程安全机制,专门用于**确保某个函数在多线程环境下只执行一次**,且是**首次调用时执行、后续调用直接返回**。它比手写双重检查锁(Double-Checked Locking)更简洁、更安全、更不易出错。

核心用法:配合使用,不可单独使用

std::once_flag 是一个标记类型,必须和 std::call_once 配合使用。它本身不提供任何成员函数,仅作为“是否已执行过”的内部状态载体。

std::call_once 是执行入口:传入一个 std::once_flag& 和一个可调用对象(函数、lambda、functor 等),它会原子地检查 flag 状态 —— 若未执行过,则执行该 callable 并标记为“已执行”;若已执行过,则直接返回,不重复执行。

基本语法示例

#include 
#include 
#include 

std::once_flag init_flag;
int global_data = 0;

void init_once() {
    std::cout << "Initializing... (thread ID: " 
              << std::this_thread::get_id() << ")\n";
    global_data = 42;
}

int main() {
    std::thread t1([]{
        std::call_once(init_flag, init_once);
    });
    std::thread t2([]{
        std::call_once(init_flag, init_once);
    });
    std::thread t3([]{
        std::call_once(init_flag, init_once);
    });

    t1.join(); t2.join(); t3.join();
    std::cout << "global_data = " << global_data << "\n"; // 输出 42
}

运行结果中,"Initializing..." 只会打印一次,无论多少个线程同时调用 std::call_once,都严格保证 init_once() 执行且仅执行一次。

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

关键注意事项

  • once_flag 必须是静态或全局生命周期:不能是局部变量(否则每次调用函数都会新建一个 flag,失去“一次”的语义)。常见做法是声明为 static std::once_flag 或全局/类静态成员。
  • call_once 是异常安全的:如果 callable 抛出异常,std::call_once 会捕获并重新抛出,且该 flag 不会被标记为“已执行”,下次调用仍会尝试执行 —— 这意味着你需确保 callable 要么成功完成,要么能处理失败重试逻辑(通常初始化函数应避免抛异常,或用 try/catch 包裹内部逻辑)。
  • 支持带参数的 callablestd::call_once(flag, func, arg1, arg2, ...) 会完美转发参数给 func(支持 move、const ref 等),无需包装成 lambda。
  • 不是互斥锁替代品:它只解决“一次性初始化”问题,不保护后续对共享数据的并发访问。初始化完成后,如需读写 global_data,仍需额外同步(如 mutex、atomic 等)。

典型应用场景

  • 单例模式的线程安全懒初始化(替代经典的 double-checked locking)
  • 全局资源首次加载(如配置文件解析、日志系统启动、GPU 上下文创建)
  • 函数内静态局部变量的底层实现机制(C++11 规定函数内 static 局部变量的初始化本身就是线程安全的,其背后常由 call_once 类似机制实现)
  • 延迟注册回调、插件初始化等“首次触发即生效”的逻辑

相关专题

更多
c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

520

2023.09.20

全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

73

2025.09.18

python 全局变量
python 全局变量

本专题整合了python中全局变量定义相关教程,阅读专题下面的文章了解更多详细内容。

96

2025.09.18

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

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

50

2025.08.29

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

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

98

2025.10.23

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

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

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

27

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外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号