当前位置: 首页 > java >正文

C++ call_once用法

🌟 概念

call_once 是 C++11 引入的并发工具,定义在 <mutex> 头文件中,用于确保某个函数在多线程环境中只被调用一次。它常用于延迟初始化(lazy initialization)或单例模式的线程安全实现。

  • 核心功能:保证一个函数在多个线程中只执行一次,即使多个线程同时尝试调用。
  • 依赖call_once 使用 std::once_flag 作为标志,确保函数的单次执行。
  • 适用场景
    • 单例对象初始化
    • 全局配置加载
    • 资源初始化(如数据库连接)

🔧 API

1. std::call_once

template <class Callable, class... Args>
void call_once(std::once_flag& flag, Callable&& func, Args&&... args);
  • 参数
    • flagstd::once_flag 对象,用于跟踪函数是否已被调用。
    • func:要执行的函数(可调用对象,如函数、lambda、函数对象等)。
    • args:传递给 func 的参数。
  • 返回值:无。
  • 行为
    • 如果 flag 表示函数未被调用,则执行 func(args...)
    • 如果 func 抛出异常,异常会传播,且 flag 状态保持未调用,允许下次重试。
    • 其他线程在 func 执行完成前会被阻塞。

2. std::once_flag

  • 定义std::once_flag 是一个非复制、非移动的类,用于与 call_once 配合。
  • 初始化:必须在定义时初始化,通常为静态或全局变量。
  • 用法:作为 call_once 的第一个参数。

⚠️ 使用注意

  • std::once_flag 不能复制或移动,必须确保同一 once_flag 对象被所有相关线程使用。
  • call_once 是线程安全的,无需额外加锁。
  • 如果 func 执行时间长,可能导致其他线程阻塞,需谨慎设计。
  • 异常安全性:若 func 抛出异常,call_once 会传播异常,且允许重试。

📦 简单示例

以下是一个使用 call_once 实现线程安全单例的例子:

#include <iostream>
#include <mutex>
#include <thread>class Singleton {
public:static Singleton& getInstance() {std::call_once(flag_, &Singleton::init);return *instance_;}void doSomething() {std::cout << "Singleton is doing something.\n";}private:Singleton() { std::cout << "Singleton constructed.\n"; }static void init() {instance_ = new Singleton();}static Singleton* instance_;static std::once_flag flag_;
};// 静态成员初始化
Singleton* Singleton::instance_ = nullptr;
std::once_flag Singleton::flag_;void worker() {Singleton& s = Singleton::getInstance();s.doSomething();
}int main() {std::thread t1(worker);std::thread t2(worker);std::thread t3(worker);t1.join();t2.join();t3.join();return 0;
}

示例输出

Singleton constructed.
Singleton is doing something.
Singleton is doing something.
Singleton is doing something.

说明

  • getInstance 使用 call_once 确保 init 只被调用一次,从而保证 Singleton 对象只构造一次。
  • 多个线程调用 getInstance 时,只有第一个线程执行 init,其他线程等待。
  • doSomething 展示了单例的使用。

✅ 优点

  • 线程安全,无需手动加锁。
  • 延迟初始化,节省资源。
  • 异常安全,支持重试。

🚫 局限性

  • 如果初始化函数耗时长,可能影响性能。
  • std::once_flag 不可复制,需小心管理。
http://www.xdnf.cn/news/13586.html

相关文章:

  • 基础知识:抽象类成员变量的继承与使用
  • Web APIS Day04
  • BlogX项目数据库读写分离AI面试题
  • 海外打车代驾app多语言切换模块设计
  • CentOs7.x系列IP地址由动态改静态(解决远程连接掉线问题)
  • 字符串|数组|计算常见函数整理-竞赛专用(从比赛真题中总结的,持续更新中)
  • 7.Vue的compute计算属性
  • atomicity of memory accesses
  • 【知识图谱构建系列1】数据集介绍
  • 本地docker部署的dify,不用git命令如何无损升级?
  • Vue3前端项目Docker容器化部署工作报告
  • 【算法 day01】LeetCode 704二分查找 | 27移除元素 | 977有序数组的平方
  • 【电力物联网】SDN架构与工作原理介绍
  • ospfOSPF特殊区域及其他特性
  • Unicode:如何让用户东方不败和[Family: Man, Woman, Girl, Boy]顺利通过用户名长度检查?
  • 【Linux指南】文件系统基础操作与路径管理
  • 爬虫+动态代理助力 AI 训练数据采集
  • [未验证]abaqus2022 更改内置python
  • 选择与方法(4) 职场内篇 沿着赤道走,到不了北极,找准职场方向,建立可迁移技能
  • 智谱的AI Agent :CoCo
  • GIS数据制备,空间分析与高级建模实践技术应用
  • 软件确认测试报告:如何评估软件功能及测试关键点?
  • 第二届“Parloo”CTF应急响应挑战赛(应急响应题目复盘)
  • ptyhon 导入本地模块 no module named Python Error几种解决方案
  • Excel文件数据的读取和处理方法——C++
  • 华为云Flexus+DeepSeek征文 | 基于华为云ModelArts Studio搭建AnythingLLM聊天助手
  • 支持在Windows电脑上使用的备忘录提醒小软件
  • 【大模型训练】中短序列attention 和MOE层并行方式
  • Java八股文——Spring「SpringBoot 篇」
  • 工业相机如何提高传输速度