0

0

C++中如何利用std::atomic_flag实现互斥访问?(轻量级锁机制)

尼克

尼克

发布时间:2026-03-11 15:53:32

|

889人浏览过

|

来源于php中文网

原创

能,std::atomic_flag可作最简自旋锁,仅支持test_and_set()和clear(),需零初始化,无递归/超时/等待能力,适合极短临界区。

c++中如何利用std::atomic_flag实现互斥访问?(轻量级锁机制)

std::atomic_flag 是不是能当锁用?

能,但只适合最简单的“开/关”场景。它本质是原子布尔标志,没有 load()store() 的语义糖,只有 test_and_set()clear() 两个操作,且默认初始化为 false(即“未设置”状态)。它比 std::mutex 轻得多,不依赖操作系统原语,纯硬件级原子指令实现——但代价是:不能递归、不能超时、不能等待,也不保证公平性。

常见错误现象:std::atomic_flag flag; 直接定义会导致未定义行为,因为默认构造不保证清零;必须显式用 ATOMIC_FLAG_INIT 或 C++20 的 std::atomic_flag{} 零初始化。

  • 正确初始化方式(C++17 及以前):std::atomic_flag flag = ATOMIC_FLAG_INIT;
  • C++20 起可直接写:std::atomic_flag flag{};(推荐)
  • 绝不能写成:std::atomic_flag flag;(未初始化,test_and_set() 行为未定义)

怎么写一个 spinlock(自旋锁)?

std::atomic_flag 实现自旋锁的核心逻辑就是:循环调用 test_and_set(),直到返回 false(说明之前是未设置状态,本次成功抢到锁);释放时调用 clear()。注意它不阻塞线程,只是忙等,所以只适用于临界区极短、且 CPU 核心数充足的场景。

性能影响明显:如果临界区稍长(比如 >100ns),自旋会浪费大量 CPU 周期,还可能因缓存行争用拖慢其他核;在单核系统上等于死锁。

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

What-the-Diff
What-the-Diff

检查请求差异,自动生成更改描述

下载
  • test_and_set() 默认带 std::memory_order_acquire 语义,进入临界区前建立内存序屏障
  • clear() 默认带 std::memory_order_release,退出时确保临界区内写入对其他线程可见
  • 若需更弱的内存序(如已用其他同步手段),可显式传参:flag.test_and_set(std::memory_order_relaxed),但极易出错,不建议初学者改

为什么不能直接用 test_and_set() 当锁判断?

因为 test_and_set() 总是把 flag 设为 true 并返回旧值——它不是“读+条件写”,而是“原子地设为 true 并告诉你原来是什么”。所以你不能写 if (flag.test_and_set()) { /* 已被占用 */ } 来判断,这会导致每次调用都强行抢占,破坏互斥逻辑。

典型误用场景:想实现“尝试获取,失败就跳过”,结果写成:

if (flag.test_and_set()) {
    // 错!这里即使失败也已把 flag 设为 true,别的线程永远拿不到锁了
    return;
}

正确做法是循环重试,或配合 clear() 配合使用:

  • 获取锁必须循环:while (flag.test_and_set(std::memory_order_acquire)) { /* 自旋 */ }
  • 释放锁必须且只能调用一次:flag.clear(std::memory_order_release);
  • 绝不能在未持有锁时调用 clear()(UB),也不能重复 clear()

和 std::mutex / std::atomic 比有什么坑?

std::atomic_flag 是唯一被要求“无锁”(lock-free)的原子类型,编译器必须生成 lock-free 汇编(如 x86 的 xchg);而 std::atomic<bool></bool> 在某些平台可能退化为内部互斥量,失去轻量优势。但正因如此,它功能极度受限:没有 operator==,不能直接取值判断,也不能用 exchange() 替代 test_and_set()

兼容性注意点:C++11 起支持,但早期 GCC/Clang 对 ATOMIC_FLAG_INIT 的宏展开有 bug,建议升级到 GCC 5.0+/Clang 3.7+;C++20 废弃该宏,统一用聚合初始化。

  • 别用 std::atomic<bool></bool> 替代:它不保证 lock-free,且多出的接口反而容易误用(比如 load() 后再 store() 不是原子的)
  • 别给 std::atomic_flagvolatile:毫无意义,原子操作本身已禁止编译器重排
  • 调试时无法打印其值(没 operator),只能靠行为推断,这是最常被忽略的调试障碍

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

846

2023.08.22

while的用法
while的用法

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

106

2023.09.25

while的用法
while的用法

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

106

2023.09.25

c++中volatile关键字的作用
c++中volatile关键字的作用

本专题整合了c++中volatile关键字的相关内容,阅读专题下面的文章了解更多详细内容。

75

2025.10.23

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1902

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

656

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2387

2025.12.29

java接口相关教程
java接口相关教程

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

47

2026.01.19

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

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

精品课程

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

共94课时 | 11.1万人学习

C 教程
C 教程

共75课时 | 5.3万人学习

C++教程
C++教程

共115课时 | 21.4万人学习

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

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