0

0

C++中的volatile关键字是什么?(为什么它不能用于线程同步)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-16 08:03:10

|

372人浏览过

|

来源于php中文网

原创

volatile不能替代mutex或atomic,因为它不提供原子性、内存屏障和执行顺序保证,仅防止编译器优化;多线程中仍会因乱序、缓存导致竞态;仅适用于硬件寄存器、信号处理等非线程场景。

c++中的volatile关键字是什么?(为什么它不能用于线程同步)

volatile 不能替代 mutex 或 atomic 的原因

它只告诉编译器“这个变量可能被外部悄悄改掉”,比如硬件寄存器、信号处理函数里的全局变量。但对多线程来说,volatile 不阻止 CPU 乱序执行,也不提供内存屏障,更不保证原子性——所以两个线程同时读写一个 volatile int,仍会崩。

常见错误现象:volatile bool flag = false; 用来做线程退出通知,主线程改 flag = true,子线程 while(!flag) 死循环,结果子线程永远看不到 true(优化或缓存导致)。

  • 它不禁止编译器把读操作提到循环外(即 hoisting)
  • 它不禁止 CPU 把写操作延迟或重排到判断之后
  • 它不保证 volatile++ 是原子的(实际是读-改-写三步)

什么场景下真的需要 volatile

主要是和硬件或系统底层打交道时:内存映射 I/O 寄存器、信号处理函数中访问的全局变量、setjmp/longjmp 跨越点修改的局部变量(需加 volatile 修饰)。

使用场景举例:嵌入式里轮询一个状态寄存器 volatile uint32_t* const STATUS_REG = (uint32_t*)0x40001000;,每次读都必须真实访存,不能被优化掉。

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

  • 信号处理函数中修改的全局变量,如 volatile sig_atomic_t g_sig_received;
  • 没有用 sigwait 而是靠全局变量通信时,必须 volatile
  • 某些旧式中断服务程序(ISR)里访问的标志位

线程安全该用什么替代 volatile

如果目标是让变量在线程间可见、修改有序、操作原子,就该用标准库提供的同步原语。

Synthesia
Synthesia

Synthesia是一个AI视频生成平台,可以让用户创建120种语言的视频。

下载

性能与兼容性影响:相比 volatilestd::atomic 在 x86 上多数操作无额外开销(如 load()/store() 默认是 memory_order_seq_cst,但编译器仍可优化成普通指令);而 mutex 开销明显,适合保护临界区,不是单个变量。

  • 单个布尔开关或计数器 → 用 std::atomic<bool></bool>std::atomic<int></int>
  • 需要保护一段逻辑(不止一个变量)→ 用 std::mutex + std::lock_guard
  • 要避免锁但又需复杂同步 → 考虑 std::atomic_flag(无锁、最轻量)

示例:std::atomic<bool> ready{false};</bool>,子线程 while(!ready.load(std::memory_order_acquire));主线程最后 ready.store(true, std::memory_order_release); —— 这才真正跨线程生效。

为什么 C++ 标准明确说 volatile 不用于线程同步

因为它的语义和并发模型根本不搭界:volatile 属于“抽象机之外的副作用”范畴,而多线程内存模型属于“抽象机之内的执行顺序与可见性”范畴。标准里连“thread”这个词都没在 volatile 描述中出现过。

容易踩的坑:有人看到 GCC 文档里说“volatile 读写有 acquire/release 效果”,就以为能凑合用——那是特定平台、特定优化等级下的巧合,不是标准行为,换 Clang 或开 -O3 就可能失效。

  • C++11 起,并发语义完全由 <atomic></atomic> 和内存序定义,volatile 未被赋予任何新含义
  • 即使在 Windows 上用 volatile 配合 Interlocked 系列函数,那也是 WinAPI 自己约定,不是 C++ 规则
  • 调试时加 volatile 可能“碰巧”让 bug 消失,但这只是掩盖了真正的竞态,不是修复

真正复杂的点在于:你得区分“谁在改这个变量”——是另一个线程?还是中断?还是硬件?答案不同,选的工具就完全不同。混用就等于没同步。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
while的用法
while的用法

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

102

2023.09.25

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

547

2023.09.20

全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

85

2025.09.18

python 全局变量
python 全局变量

本专题整合了python中全局变量定义相关教程,阅读专题下面的文章了解更多详细内容。

101

2025.09.18

全局变量怎么定义
全局变量怎么定义

本专题整合了全局变量相关内容,阅读专题下面的文章了解更多详细内容。

85

2025.09.18

python 全局变量
python 全局变量

本专题整合了python中全局变量定义相关教程,阅读专题下面的文章了解更多详细内容。

101

2025.09.18

string转int
string转int

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

750

2023.08.02

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

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

568

2024.08.29

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

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

145

2026.02.13

热门下载

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

精品课程

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

共94课时 | 9.5万人学习

C 教程
C 教程

共75课时 | 4.7万人学习

C++教程
C++教程

共115课时 | 17.9万人学习

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

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