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

【C/C++】无锁编程——compare_exchange_strong

文章目录

  • 原子操作之compare_exchange_strong
    • 1 简介
    • 2 核心
    • 2 语法
    • 3 与 compare_exchange_weak的区别
    • 4 使用场景示例
      • 4.1 无锁计数器
      • 4.2 线程安全栈(Treiber Stack)
    • 5 内存序(Memory Order)参数
    • 6 关键注意事项
    • 7 总结

原子操作之compare_exchange_strong

1 简介

std::atomic 类型提供的一个关键原子操作.
作用:

  1. 实现无锁(lock-free)编程中的 比较并交换(Compare-and-Swap, CAS)
  2. 确保在多线程环境中安全地修改共享数据,避免竞态条件(race condition)。

2 核心

  1. 比较并交换

    • 输入参数
      • expected:期望的旧值(引用传递,操作后可能被修改)。
      • desired:希望设置的新值。
      • memory_order:内存序(可选,默认 std::memory_order_seq_cst)。
    • 操作逻辑
      • 如果原子变量的当前值等于 expected,则将其设置为 desired,并返回 true
      • 如果当前值不等于 expected,则将 expected 更新为当前值,并返回 false
  2. 原子性保证

    • 整个操作是原子的,不会被其他线程中断。

2 语法

bool compare_exchange_strong(T& expected, T desired,std::memory_order order = std::memory_order_seq_cst
);

3 与 compare_exchange_weak的区别

特性compare_exchange_strongcompare_exchange_weak
伪失败永远不会伪失败(总是严格的 CAS)允许在某些平台返回伪失败(即使值匹配)
适用场景需要确保操作成功的一次性场景循环中重试的场景(如自旋锁)
性能可能略低(需严格检查)可能更高(允许硬件优化)
典型代码模式单次检查(如无锁队列的精确条件)循环重试(如 while (!weak(...))

4 使用场景示例

4.1 无锁计数器

std::atomic<int> counter{0};void increment() {int old_val = counter.load();while (true) {int new_val = old_val + 1;if (counter.compare_exchange_strong(old_val, new_val)) {break; // 成功递增}// 失败时 old_val 已更新为当前值,继续重试}
}

4.2 线程安全栈(Treiber Stack)

template<typename T>
class LockFreeStack {struct Node {T data;Node* next;};std::atomic<Node*> head{nullptr};public:void push(const T& value) {Node* new_node = new Node{value, head.load()};while (!head.compare_exchange_strong(new_node->next, new_node));}bool pop(T& value) {Node* old_head = head.load();while (old_head && !head.compare_exchange_strong(old_head, old_head->next)) {}if (!old_head) return false;value = old_head->data;delete old_head;return true;}
};

5 内存序(Memory Order)参数

compare_exchange_strong 接受以下内存序(控制操作的原子性和可见性):

  • std::memory_order_relaxed:无同步(仅保证原子性)。
  • std::memory_order_acquire:当前线程的后续读操作必须在此操作后执行。
  • std::memory_order_release:当前线程的前序写操作必须在此操作前完成。
  • std::memory_order_acq_rel:结合 acquirerelease
  • std::memory_order_seq_cst:全局顺序一致性(默认,性能最低)。

6 关键注意事项

  1. ABA 问题

    • 如果 expected 值在操作期间被其他线程修改为其他值后又改回原值,compare_exchange_strong 会误认为值未变。
    • 解决方案:使用带版本号的原子变量(如 std::atomic<std::pair<T, uint64_t>>)或 hazard pointer
  2. 循环重试

    • 在冲突激烈时,compare_exchange_strong 可能导致忙等待(busy-wait),需结合退避策略(如指数退避)。
  3. 性能优化

    • 在循环中使用 compare_exchange_weak 更高效(如自旋锁),而 compare_exchange_strong 适合单次检查。

7 总结

compare_exchange_strong 是 C++ 无锁编程的核心工具,通过严格的 CAS 语义保证线程安全。使用时需结合内存序参数、循环重试机制,并警惕 ABA 问题。在需要精确条件判断时优先使用 strong 版本,而在循环中可优先使用 weak 版本以提升性能。

相关文章:

  • LeetCode 热题 100 46. 全排列
  • 为React组件库引入自动化测试:从零到完善的实践之路
  • 【CF】Day51——Codeforces Round 963 (Div. 2) CD
  • 【AI学习】DeepSeek-R1是如何训练的?
  • 我的世界Minecraft游戏服务器搭建教程:腾讯云Java版
  • 学习黑客Nmap 原理
  • 时间同步服务核心知识笔记:原理、配置
  • 【信息系统项目管理师-论文真题】2006下半年论文详解(包括解题思路和写作要点)
  • C# 检查某个点是否存在于圆扇区内(Check whether a point exists in circle sector or not)
  • 五大神经网络开发实战:从入门到企业级部署
  • 【数据结构与算法】同余计算 哈希表与前缀和问题特征和模板化思路
  • 滚珠螺杆的精度如何保持?
  • Nacos源码—3.Nacos集群高可用分析二
  • Vue中的过滤器参数:灵活处理文本格式化
  • Docker 使用下 (二)
  • 知识图谱 + 大语言模型:打造更聪明、更可靠的AI大脑 —— 探索 GraphRAG 中文优化与可视化实践
  • VirtualBox调整虚拟机内存和CPU
  • 数据库的原子事务
  • 阿里云物联网平台--云产品流传
  • Qt6.8中进行PDF文件读取和编辑
  • 山东滕州一车辆撞向公交站台撞倒多人,肇事者被控制,案件已移交刑警
  • “五一”假期预计全社会跨区域人员流动累计14.67亿人次
  • 媒体:南京秦淮区卫健委回应一医院涉嫌违规提供试管婴儿服务
  • 抚州一原副县长拉拢公职人员组建“吃喝圈”,长期接受打牌掼蛋等“保姆式”服务
  • 原油价格战一触即发?沙特不想再忍,领衔多个产油国加速增产
  • 来上海喝云南咖啡!上海国际咖啡文化节助力咖啡产业破圈出海