c++怎么利用std::call_once确保初始化一次_c++ 多线程环境单例安全加载【方法】
发布时间 - 2026-01-09 00:00:00 点击率:次std::call_once能保证只执行一次,因其内部采用原子操作加互斥锁双重机制,确保多线程下仅一个线程执行可调用对象,其余阻塞等待;正确使用需满足三条件:once_flag须为静态存储期、可调用对象不可抛异常、多线程共享同一flag实例。
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 Sin
gleton();
});
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_ 必须定义在类外,否则链接失败。
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 —— 这种问题不会报错,但行为未定义。
# 栈
# c++
# 区别
# 为什么
# Static
# 构造函数
# 局部变量
# Lambda
# 指针
# 线程
# 多线程
# 并发
# 对象
# 多个
# 首次
# 死锁
# 的是
# 都有
# 放在
# 互斥
# 都能
# 已有
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Windows10电脑怎么设置虚拟光驱_Win10右键装载ISO镜像文件
网页制作模板网站推荐,网页设计海报之类的素材哪里好?
在centOS 7安装mysql 5.7的详细教程
如何用花生壳三步快速搭建专属网站?
javascript和jQuery中的AJAX技术详解【包含AJAX各种跨域技术】
青岛网站建设如何选择本地服务器?
零基础网站服务器架设实战:轻量应用与域名解析配置指南
Python企业级消息系统教程_KafkaRabbitMQ高并发应用
,网页ppt怎么弄成自己的ppt?
Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程
Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案
Laravel如何使用查询构建器?(Query Builder高级用法)
Laravel定时任务怎么设置_Laravel Crontab调度器配置
Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比
如何用狗爹虚拟主机快速搭建网站?
香港服务器WordPress建站指南:SEO优化与高效部署策略
Laravel队列由Redis驱动怎么配置_Laravel Redis队列使用教程
微信h5制作网站有哪些,免费微信H5页面制作工具?
如何在建站之星绑定自定义域名?
免费网站制作appp,免费制作app哪个平台好?
网站制作报价单模板图片,小松挖机官方网站报价?
laravel怎么用DB facade执行原生SQL查询_laravel DB facade原生SQL执行方法
如何在阿里云虚拟机上搭建网站?步骤解析与避坑指南
如何在万网自助建站平台快速创建网站?
用v-html解决Vue.js渲染中html标签不被解析的问题
如何快速搭建二级域名独立网站?
linux写shell需要注意的问题(必看)
Laravel N+1查询问题如何解决_Eloquent预加载(Eager Loading)优化数据库查询
Laravel如何处理CORS跨域请求?(配置示例)
Laravel Debugbar怎么安装_Laravel调试工具栏配置指南
免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?
PHP的CURL方法curl_setopt()函数案例介绍(抓取网页,POST数据)
Laravel如何实现用户角色和权限系统_Laravel角色权限管理机制
html5怎么画眼睛_HT5用Canvas或SVG画眼球瞳孔加JS控制动态【绘制】
JavaScript中如何操作剪贴板_ClipboardAPI怎么用
JS中使用new Date(str)创建时间对象不兼容firefox和ie的解决方法(两种)
如何在云主机快速搭建网站站点?
米侠浏览器网页图片不显示怎么办 米侠图片加载修复
Firefox Developer Edition开发者版本入口
Laravel怎么实现验证码功能_Laravel集成验证码库防止机器人注册
详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)
如何快速启动建站代理加盟业务?
Laravel怎么使用artisan命令缓存配置和视图
清除minerd进程的简单方法
Laravel DB事务怎么使用_Laravel数据库事务回滚操作
android nfc常用标签读取总结
如何在阿里云通过域名搭建网站?
Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例
,怎么在广州志愿者网站注册?
如何快速使用云服务器搭建个人网站?


gleton();
});
return *instance_;
}