volatile 关键字核心作用

一句话总结:阻止编译器对该变量做缓存优化,强制每次读写都访问原始内存,不从CPU寄存器拿缓存值

注意:volatile 不是优化代码,是禁用编译器优化

1、底层原理

不加volatile: 编译器为提速,变量读一次后存入CPU寄存器 ,后续重复取值直接用寄存器缓存,不再访问内存。 加volatile: 编译器被强制:每次读=去内存地址取值;每次写=立刻把数据写回内存,不许寄存缓存

2、三大使用场景(嵌入式C最常用,STM32必备)

① 硬件外设寄存器(最常用)

外设寄存器的值由硬件自动变化(引脚、外设),不加volatile编译器会缓存旧值,读不到硬件实时状态。

复制代码
// 寄存器地址强制volatile,保证每次读写真实硬件地址
#define PA_MOD (*(volatile unsigned int *)0x40020000)

② 中断修改的全局变量

中断函数修改全局标记,主函数循环查询;无volatile编译器优化后,主循环永远读取寄存器旧值,卡死死循环。

复制代码
volatile uint8_t rec_flag = 0; // 中断里置1
while(rec_flag == 0);         // 必须volatile才能感知中断修改

③ 多任务/多线程共享变量

一个任务修改变量、另一个任务读取,防止编译器缓存导致读取数据滞后。

3、关键误区

  1. 不能保证原子操作:volatile无法防止多线程同时改写数据,不能替代互斥锁;

  2. 不优化性能,反而降速:频繁访问内存比寄存器慢,只在必要时添加;

  3. 不屏蔽CPU硬件Cache:只管编译器优化,管不了CPU高速缓存。

4、结合你截图代码

复制代码
int add(volatile int a, volatile int b)
{
    volatile int sum;
    sum = a + b;
    return sum;
}

全部加volatile: 每次取a、b都读内存,结果存入sum必须写内存,return再次读sum内存; 去掉volatile后O2优化会精简代码、省略内存存取,甚至直接合并成一条加法指令。

需要我对比有无volatile的汇编代码直观看出区别吗?

相关推荐
Dxy12393102161 小时前
Python Tensor 向量入门:从零理解深度学习的“数据语言“
开发语言·python·深度学习
林森lsjs1 小时前
【日耕一题】3. 通过键盘输入一行字符,分别统计出其中英文字母、空格、数字和其它字符的个数。
java·开发语言
yzy852 小时前
数据同步工具 -- syncthing
开发语言
catchadmin2 小时前
PHP 应用 security.txt 漏洞披露实践
开发语言·php
糖果店的幽灵2 小时前
LangChain 1.3 完全教程:从入门到精通-Part 11: Tools(工具系统)
开发语言·langchain·c#
夜勤月2 小时前
AQS 与 ThreadPoolExecutor 深度拆解:JDK 高并发底层设计精髓
android·java·开发语言
luj_17682 小时前
R语言生态优势与学习曲线分析
c语言·开发语言·网络·经验分享·算法
程序大视界2 小时前
【C++ 从基础到项目实战】C++(二):数组、字符串与结构体——组织数据的容器
开发语言·c++·cpp
叶子野格2 小时前
《C语言学习:文件操作》16
c语言·开发语言·c++·学习·visual studio