0

0

c++中如何实现一个高性能的无锁队列? (CAS操作详解)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-14 11:43:15

|

617人浏览过

|

来源于php中文网

原创

直接用 std::atomic 实现无锁队列易出错,主因是mpmc场景下单指针cas无法保证插入/弹出原子性,且面临aba问题;安全方案需用带版本号的tagged pointer(如48位指针+16位版本号),要求指针64kb对齐,并严格配对memory_order。

c++中如何实现一个高性能的无锁队列? (cas操作详解)

为什么直接用 std::atomic 实现无锁队列容易出错?

因为无锁队列本质是多生产者多消费者(MPMC)场景,仅靠对 headtail 单个指针做 fetch_add 无法保证节点插入/弹出的原子性。典型错误是「ABA问题」:线程A读到 tail == p,被调度走;线程B把 p 弹出又重用了同一内存地址;线程A回来继续 CAS,误以为链表结构没变,导致指针断裂或重复释放。

真正安全的做法必须同时控制两个指针状态,常见方案是使用双字 CAS(double-word CAS),但 x86-64 的 cmpxchg16b 指令需开启特定编译选项且部分老 CPU 不支持;更通用的解法是引入版本号(tagged pointer)——把指针低几位腾出来存计数,避免 ABA。

std::atomic<uint64_t></uint64_t> 模拟 tagged pointer 的写法

将 64 位整数拆成 48 位指针 + 16 位版本号(足够应对绝大多数场景),所有 CAS 都作用于这个整数。关键点在于:指针必须 64KB 对齐(确保低 16 位为 0),才能安全移位。

  • 分配节点时用 aligned_alloc(65536, sizeof(Node)) 或自定义内存池
  • 读取指针:ptr = (Node*)(val & ~0xFFFFULL)
  • 读取版本:ver = val & 0xFFFFULL
  • CAS 前必须构造新值:new_val = ((uint64_t)new_ptr) | ((old_ver + 1) & 0xFFFFULL)
struct Node {
    int data;
    std::atomic<uint64_t> next; // 存储 tagged pointer
};

struct LockFreeQueue {
    std::atomic<uint64_t> head; // tagged pointer
    std::atomic<uint64_t> tail;

    LockFreeQueue() : head(0), tail(0) {}
};

CAS 循环中必须检查空队列和满队列边界

无锁队列不是“无边界”,而是边界检查必须嵌入 CAS 循环内,否则多个线程可能同时判定“可入队”却争抢同一槽位。例如基于数组的环形无锁队列(如 Dmitry Vyukov 队列),enqueue 中要反复读 tail、计算索引、检查 queue[(tail + 1) % capacity] != nullptr,再尝试 CAS 更新 tail —— 这个检查不能放在 CAS 外部,否则会漏掉其他线程刚写入的值。

SekoTalk
SekoTalk

商汤科技推出的AI对口型视频创作工具

下载

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

基于链表的实现同样如此:插入前必须确认 tail->next == nullptr,但这个判断本身不是原子的,所以要用 CAS 尝试设置 tail->next = new_node,失败则说明已有其他线程抢先推进了 tail,需重新读取最新 tail

实际项目中更推荐用成熟实现而非手写

Linux 内核的 lockless list、Facebook 的 Folly::MPMCQueue、Boost.Lockfree,都经过大量压测和内存模型验证。手写无锁结构最易出问题的地方不在 CAS 本身,而在内存序(memory order)选择:std::memory_order_acquirestd::memory_order_release 必须成对出现在读/写路径上,漏掉一个就可能导致重排序后看到脏数据;而 std::memory_order_relaxed 在某些路径下看似能提升性能,实则破坏 happens-before 关系。

如果你只是需要高吞吐队列,优先考虑 Folly::MPMCQueue 并启用 cache_line_size 对齐;若必须手写,请用 clang++ -fsanitize=thread 跑压力测试,而不是靠逻辑推演确认正确性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

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

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

294

2025.08.29

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

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

105

2025.10.23

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

763

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

98

2025.12.01

磁盘配额是什么
磁盘配额是什么

磁盘配额是计算机中指定磁盘的储存限制,就是管理员可以为用户所能使用的磁盘空间进行配额限制,每一用户只能使用最大配额范围内的磁盘空间。php中文网为大家提供各种磁盘配额相关的内容,教程,供大家免费下载安装。

1542

2023.06.21

如何安装LINUX
如何安装LINUX

本站专题提供如何安装LINUX的相关教程文章,还有相关的下载、课程,大家可以免费体验。

716

2023.06.29

linux find
linux find

find是linux命令,它将档案系统内符合 expression 的档案列出来。可以指要档案的名称、类别、时间、大小、权限等不同资讯的组合,只有完全相符的才会被列出来。find根据下列规则判断 path 和 expression,在命令列上第一个 - ( ) , ! 之前的部分为 path,之后的是 expression。还有指DOS 命令 find,Excel 函数 find等。本站专题提供linux find相关教程文章,还有相关

300

2023.06.30

linux修改文件名
linux修改文件名

本专题为大家提供linux修改文件名相关的文章,这些文章可以帮助用户快速轻松地完成文件名的修改工作,大家可以免费体验。

799

2023.07.05

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

44

2026.03.06

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 10.3万人学习

Git 教程
Git 教程

共21课时 | 4.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号