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的汇编代码直观看出区别吗?

相关推荐
LDR0067 天前
Type-C 快充全面升级!LDR6601 赋能个人护理便携电机,重塑剃须刀 / 理发器新体验
c语言·开发语言
雪碧聊技术7 天前
Tree.js是什么?一文讲透
开发语言·javascript·ecmascript
码云数智-园园7 天前
C++20 Modules 模块详解
java·开发语言·spring
swordbob7 天前
NIO的channel中什么是 fd(File Descriptor,文件描述符)
java·开发语言·nio
源分享7 天前
Java线程同步的多种实现方法(非常详细)
java·开发语言·jvm
Luminous.7 天前
C语言--day30
c语言·开发语言
何以解忧,唯有..7 天前
Go语言循环语句详解:for、range与循环控制
开发语言·算法·golang
謓泽7 天前
C语言不是语法,是通往机器的地图。
c语言·开发语言
云水一下7 天前
从零开始学 PHP 系列(一):PHP 的前世今生与开发环境搭建
开发语言·php
飞天狗1117 天前
零基础JavaWeb入门——第五课第二小节:九大内置对象 · 第2个:response(响应对象)
java·开发语言