0

0

C++的std::atomic_flag如何实现一个超轻量级的自旋锁? (无锁编程实战)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-20 10:56:03

|

755人浏览过

|

来源于php中文网

原创

std::atomic_flag比std::atomic更适合自旋锁,因其唯一保证无锁、直接映射cpu原子指令;而std::atomic在部分平台可能退化为加锁实现,且不支持test_and_set()。

c++的std::atomic_flag如何实现一个超轻量级的自旋锁? (无锁编程实战)

std::atomic_flag 为什么比 std::atomic 更适合做自旋锁

因为 std::atomic_flag 是 C++ 标准里唯一保证「无锁(lock-free)」的原子类型,底层可直接映射到 CPU 的 TSETLDXR/STXRXCHG 指令;而 std::atomic<bool></bool> 在某些平台(比如旧 ARM 或某些嵌入式编译器)可能被实现为加锁的库函数——一旦锁住,就不是自旋锁了,而是退化成阻塞等待。

常见错误现象:std::atomic<bool> test{true}; test.test_and_set()</bool> 编译失败,因为 test_and_set() 不是 std::atomic<bool></bool> 的成员函数——它只属于 std::atomic_flag

  • std::atomic_flag 默认初始化为 clear(即 false),必须用 ATOMIC_FLAG_INIT(C++17 起已弃用,推荐用 constexpr 构造)
  • 不支持直接赋值或比较,只能靠 test_and_set()clear()
  • 没有 load()/store(),所以没法“只读检查是否被占用”,必须尝试获取

怎么写一个可用的自旋锁类(带 memory_order 控制)

核心就是两步:循环调用 test_and_set() 直到成功,临界区结束后调用 clear()。但关键在内存序——默认 test_and_set()memory_order_seq_cst,太重;实际只需 acquire/release 语义即可。

使用场景:短临界区(比如保护几个指针赋值、计数器增减)、高争用但平均持有时间极短、不允许线程休眠(实时系统、中断上下文模拟)。

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

Flux AI
Flux AI

Flux AI,释放你的想象力,用文字生成图像

下载
  • 构造时用 constexpr std::atomic_flag flag{ATOMIC_FLAG_INIT}(C++20 起可直接 std::atomic_flag flag{};
  • lock() 中用 flag.test_and_set(std::memory_order_acquire) —— 成功时返回旧值(false),失败返回 true,需循环
  • unlock() 必须用 flag.clear(std::memory_order_release),否则其他线程看不到临界区内的写操作
  • 别忘了在析构中 clear(),否则若对象被销毁时仍处于 set 状态,后续复用会死锁
class spinlock {
    std::atomic_flag flag = ATOMIC_FLAG_INIT;
public:
    void lock() {
        while (flag.test_and_set(std::memory_order_acquire)) {
            // 可选:__builtin_ia32_pause() 或 std::this_thread::yield()
        }
    }
    void unlock() {
        flag.clear(std::memory_order_release);
    }
};

为什么空循环自旋不是最优?怎么微调提示 CPU

while(flag.test_and_set(...)) {} 在 x86 上会持续发射 XCHG,不仅耗电,还可能让其他超线程核心饿死;ARM 上更糟,可能触发总线风暴。这不是“轻量”,是“暴力轮询”。

性能影响:在争用激烈但临界区极短的场景下,加 pause/yield 后,实测吞吐能提升 10%~30%,cache miss 减少明显。

  • x86/x64:在循环体内插一句 __builtin_ia32_pause()(GCC/Clang),或 _mm_pause()(需 <immintrin.h></immintrin.h>
  • 通用可移植写法:用 std::this_thread::yield(),但注意它不保证暂停,只是提示调度器,开销比 pause 大得多
  • 别用 std::this_thread::sleep_for(1ns)——这是系统调用,完全违背“超轻量”初衷

std::atomic_flag 自旋锁的致命限制和替代思路

它不能递归、不能超时、不感知线程取消、也不通知等待者。一旦临界区里发生 longjmp、异常未捕获、或死循环,整个程序就卡死——因为没地方能“中断”自旋。

容易被忽略的地方:std::atomic_flag 对象若位于多线程共享的全局缓存行(cache line)上,且附近有高频修改的变量,会引发 false sharing。建议用 alignas(64) 对齐,并确保前后无其他热数据。

  • 需要超时?得自己实现带计数的重试 + std::chrono 判断,但已脱离“超轻量”范畴
  • 要支持中断或取消?换 std::mutex,哪怕它重,也比卡死强
  • 临界区超过 10 条指令、或含系统调用(如 read()malloc)?立刻放弃自旋锁——CPU 在那干等毫无意义

真要用,就把它当成“临界区是几条 mov + add 的硬件级临界保护”,别的都别指望。

相关文章

编程速学教程(入门课程)
编程速学教程(入门课程)

编程怎么学习?编程怎么入门?编程在哪学?编程怎么学才快?不用担心,这里为大家提供了编程速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

103

2023.09.25

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

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

675

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

369

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

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

26

2026.01.21

C++多线程相关合集
C++多线程相关合集

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

24

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

99

2026.02.06

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

660

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

203

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

95

2026.02.13

热门下载

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

精品课程

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

共94课时 | 9.8万人学习

C 教程
C 教程

共75课时 | 4.8万人学习

C++教程
C++教程

共115课时 | 18.5万人学习

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

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