0

0

C++ volatile能保证原子性吗 C++ 多线程内存可见性误区【并发】

冰火之心

冰火之心

发布时间:2026-01-29 10:45:12

|

529人浏览过

|

来源于php中文网

原创

volatile不能保证原子性或线程间可见性,仅禁用编译器优化;多线程同步必须使用std::atomic配合内存序或互斥锁。

c++ volatile能保证原子性吗 c++ 多线程内存可见性误区【并发】

volatile 不能保证原子性,连基本的读-改-写都可能出错

volatile 在 C++ 中只影响编译器优化行为,禁止对变量的读写被重排或省略,但它**不生成任何内存屏障(memory barrier)指令**,也不阻止 CPU 级别的乱序执行。这意味着:i++(即 read-modify-write)这种操作在 volatile int i 上依然是非原子的——多个线程同时执行它,结果大概率丢失更新。

  • 典型错误:用 volatile bool flag 做线程间“信号量”,以为设为 true 后另一线程立刻能读到新值并退出循环
  • 实际风险:即使值被写入,由于缺乏 acquire/release 语义,读线程可能永远看不到更新(尤其在 ARM/PowerPC 上),或看到中间态(如结构体部分更新)
  • 汇编层面:GCC/Clang 对 volatile 只加 mov 指令,不会插入 ldarstlrmfence

内存可见性靠的是 memory_order,不是 volatile

多线程下变量修改对其他线程“可见”,依赖的是 C++11 引入的内存序模型,而非 volatile。例如:

std::atomic ready{false};
// 线程 A
data = 42;
ready.store(true, std::memory_order_release); // 保证 data 写入对 B 可见

// 线程 B while (!ready.load(std::memory_order_acquire)) { } // 等待 assert(data == 42); // 成立

这里起作用的是 memory_order_releasememory_order_acquire 构成的同步关系,volatile 完全无法提供等价保障。

  • volatile 不参与数据竞争定义:C++ 标准明确将 volatile 访问排除在“数据竞争”检查之外
  • 调试时可能“碰巧”看到可见性:因为编译器禁用了优化,且频繁读写触发了缓存同步,但这不可靠、不可移植
  • 混合使用 volatile + atomic 没有意义,反而干扰语义清晰性

什么场景下还能用 volatile?

仅限于与硬件寄存器、信号处理函数、或某些嵌入式裸机环境交互时,需要防止编译器擅自优化掉必须发生的读写。例如:

知了zKnown
知了zKnown

知了zKnown:致力于信息降噪 / 阅读提效的个人知识助手。

下载

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

volatile uint32_t* const UART_REG = reinterpret_cast(0x40001000);
*UART_REG = 0xFF; // 强制写入,不能被优化掉
  • 信号处理:全局 volatile sig_atomic_t flag 是 POSIX 允许在线程/信号上下文中安全访问的极少数类型之一
  • 单线程嵌入式轮询:比如等待某个外设状态位变 1,且确认无其他线程/中断修改该地址
  • 绝不能用于:线程间通信、锁实现、引用计数、状态机标志(除非配合 atomic 或 mutex)

替代方案:用 std::atomic 替代 volatile 的常见误用

多数想用 volatile 解决“变量变化要立刻被看到”的地方,真正该用的是 std::atomic区别在于:

  • std::atomic 默认提供 memory_order_seq_cst,既保证原子性,也提供跨线程顺序保证
  • 若需性能,可降级为 memory_order_relaxed(仅保原子性)、memory_order_acquire(仅保读可见性)等
  • 注意:std::atomic_flag 是唯一无锁(lock-free)且保证无内存顺序开销的类型,适合自旋锁底层
  • 结构体不能直接 atomic,除非是 trivially copyable 且满足对齐要求;否则得用 std::atomic<:shared_ptr>> 或互斥保护

真正棘手的地方不在语法,而在于:很多人把 “volatile 修饰了” 当作“线程安全”的心理暗示,但 C++ 并发模型里,没有显式同步原语(atomicmutexcondition_variable)参与的共享访问,就是未定义行为。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

220

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

192

2025.07.04

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

220

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

192

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

463

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

544

2024.08.29

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

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

93

2025.08.29

C++中int的含义
C++中int的含义

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

197

2025.08.29

Golang 网络安全与加密实战
Golang 网络安全与加密实战

本专题系统讲解 Golang 在网络安全与加密技术中的应用,包括对称加密与非对称加密(AES、RSA)、哈希与数字签名、JWT身份认证、SSL/TLS 安全通信、常见网络攻击防范(如SQL注入、XSS、CSRF)及其防护措施。通过实战案例,帮助学习者掌握 如何使用 Go 语言保障网络通信的安全性,保护用户数据与隐私。

0

2026.01.29

热门下载

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

精品课程

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

共94课时 | 7.9万人学习

C 教程
C 教程

共75课时 | 4.3万人学习

C++教程
C++教程

共115课时 | 14.5万人学习

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

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