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

在C++多线程编程中,某些资源或操作只需要初始化一次,但可能被多个线程同时尝试执行。为了确保这类初始化操作只运行一次且线程安全,std::call_once 提供了一种简洁可靠的机制。
std::call_once 是 C++11 引入的工具,定义在 <mutex> 头文件中。它与 std::once_flag 配合使用,保证某个可调用对象(如函数、lambda)在整个程序生命周期中仅被执行一次,即使被多个线程并发调用。
其函数原型为:
template<class Callable, class... Args>
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<Singleton> instance;
};
std::once_flag Singleton::initFlag;
std::unique_ptr<Singleton> Singleton::instance;
多个线程同时调用 getInstance() 时,构造函数仅执行一次,无需手动加锁判断。
有些全局服务(如日志系统、配置加载、信号处理注册)只需初始化一次。使用 std::call_once 可以避免重复初始化和数据竞争。
例如:
std::once_flag logInitFlag;
void initLogging() {
std::call_once(logInitFlag, [](){
// 打开日志文件、设置格式等
std::cout << "Logging initialized.\n";
});
}
各模块均可安全调用 initLogging(),实际初始化只发生一次。
过去开发者可能用 volatile bool 标记 + 互斥锁实现“一次执行”,但容易遗漏内存序或死锁。std::call_once 内部已处理所有同步细节,包括异常安全:如果初始化函数抛出异常,call_once 会认为本次调用失败,允许下一次尝试再次执行(直到成功为止)。
这意味着:
这种行为非常适合容错性要求高的初始化流程。
基本上就这些。std::call_once 看似简单,但在构建健壮的多线程程序时非常实用,能有效消除竞态条件,让一次性初始化变得清晰又安全。
以上就是c++++中std::call_once的使用场景_c++线程安全的单次初始化机制讲解的详细内容,更多请关注php中文网其它相关文章!
c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号