历史上的今天
返回首页

历史上的今天

今天是:2025年12月15日(星期一)

正在发生

2022年12月15日 | 单片机volatile关键字的作用

2022-12-15 来源:zhihu

最近有些想伙伴问在单片机c语言编程时volatile这个关键词怎么理解?有什么作用?

Volatile是C语言的一个关键字,在stm8和stm32的固件库里也会经常看见这个关键字的使用。


如果真的想理解透彻,可能要追溯到编译器优化机制和cpu访问变量的原理。

这个可能需要花很多时间去研究,并不是很划算,我一直强调学习也要把控好投产比,也就是你投入研究的时间要和回报成正比。


很明显,对于一个开发者来说并不需要深入研究编译器优化代码的机制,至少我这么多年没深入研究过,也不影响我做产品。


很熟悉和只了解,同样都能实现功能,所花的时间成本却是天差地别的。

所以,我们只需要知道有编译器优化的概念,和cpu访问变量的原理就够了。

重点是记住volatile这个关键词的使用场景,什么时候必须要用!


一. CPU访问变量

1. 给变量赋值

我们看上图代码。

代码第一行定义了一个全局变量a,第6行把a赋值为1。

程序在执行的时候,首先会把1这个值赋值给cpu的寄存器里(比如R0-R7),然后再把寄存器里面这个1赋值到&a这个内存地址里。

这是给变量赋值的过程。

2. 读变量的值

代码第一行定义了两个全局变量a和b,第12行把变量a的值赋值给b。

程序在执行的时候,会先把a这个内存地址的值(也就是1)取出来先存到寄存器里,然后再把寄存器里的值存储到变量b的内存地址里。

这是读取变量值的过程。

二.编译器优化

1.上面程序执行过程有什么缺点?

我们可以再看一下上面那个代码,是不是觉得程序执行过程中第16行(&a内存地址->寄存器)的操作有点多余?

因为a的值已经存在寄存器里了,直接把寄存器的值存到变量b的内存地址里不就行了?

为什么还要重新从a的内存地址里重新取值到寄存器?有种脱裤子放屁的感觉对吧?

2.为什么要优化?

你能想到的,大聪明编译器肯定也能想到。

一般访问寄存器要比访问内存(RAM)的效率高。

所以,编译器也是基于这些规律特点,对我们写好的代码进行优化。

我们常用的开发工具Keil可以对优化等级进行设置,比如说Keil这个开发工具可以设置优化等级。

具体这几个优化等级有啥区别,大家可以自行百度下。

经过编译器代码优化以后,为了执行效率更高,执行这段程序的流程就会被优化。

最终程序执行可能就直接把寄存器的值赋值给变量b这个内存地址了,而不是重新从变量a的内存地址里读取到寄存器,这样效率就提高了。

编译器优化原则之一:减少对内存访问的次数,因为从内存里读写数据效率比较低。

三.volatile到底有什么用?用在什么场合?

如果这样被优化掉,会不会出现问题?哪些情况下会出现问题?

答案是有可能会,比如以下几种情况:

1.在执行b=a之前,发生中断,中断里把a的值改了

假设程序执行到第12行,发生了一个定时器4中断,中断里a=2了。

这个时候b=a,大家猜结果等于多少?是不是还是等于1?

而实际上a=2了,这个时候程序就会产生一些不可预知的错误。

所以,如果你有全局变量,会在中断里去改变它的值,最好用volatile关键字修饰下。

2.带RTOS的情况下

相信大家都知道,实时操作系统任务之间是可以根据任务优先级打断的。

如上图,假设Task2任务优先级比Task1优先级高,说明Task2是可以随时打断Task1的。

假设Task1执行到12行,Task2任务就绪开始执行了,然后把a=2,执行完又回到Task1的第14行继续执行,这样b=a,大家猜猜结果等于多少?是不是还是等于1?

实际上a=2了,这个时候程序也会产生一些不可预知的错误。

所以,如果你程序加了RTOS,并且多个任务共享一个全局变量时,最好也用volatile关键词修饰下。

3.变量读取单片机寄存器值时

USART1->DR是STM32单片机串口1的数据寄存器,当有串口数据发送和接收的时候,数据都会在这个寄存器里。

USART1->DR里面的数据可能会一直会变化的。

假设cpu执行到上图程序第9-13行时,USART1->DR发生改变,那a和b的值肯定也不相等。

所以,如果你有变量是读取寄存器的值,最好也用volatile关键词修饰下。


四.最后总结

简单来说,volatile就是告诉编译器编译时不要过渡优化,以便告诉cpu,当你执行到这个变量的时候,记得重新从内存里去读,以保证读出来的值是最新的。

一般以下使用场景需要用:

1.中断里会改变全局变量的值

2.多任务共享同一全局变量

3.变量读单片机寄存器值


推荐阅读

史海拾趣

Electromagnetic Industries Llp公司的发展小趣事

进入21世纪,随着科技的不断进步,电子行业对产品的性能和质量要求越来越高。EMI公司意识到,只有不断创新才能在激烈的市场竞争中立于不败之地。因此,公司加大了对研发的投入,引进了一批高端技术人才,并建立了完善的研发体系。经过几年的努力,EMI公司成功研发出了一系列具有自主知识产权的电磁产品,这些产品在性能和质量上均达到了国际先进水平,赢得了客户的广泛赞誉。

Avel Lindberg公司的发展小趣事

EMI公司深知产品质量对企业的重要性。因此,公司建立了严格的质量管理体系,从原材料采购到生产加工再到产品检验,每一个环节都严格把关。同时,公司还引进了先进的检测设备和技术手段,确保产品质量的稳定性和可靠性。这些措施的实施使得EMI公司的产品质量得到了客户的广泛认可。

意华(CZT)公司的发展小趣事

意华在发展过程中,积极拓展国内外市场,与华为、中兴、富士康等知名企业建立了长期稳定的合作伙伴关系。这些合作伙伴的加入,不仅为意华带来了更多的订单和业务机会,也促进了公司技术的交流和提升。通过与这些行业巨头的合作,意华在电子连接器领域的影响力逐渐扩大。

CONEC公司的发展小趣事

在电子行业中,技术的突破和创新是企业持续发展的关键。CONEC深知这一点,因此不断投入研发资源,力求在技术上取得突破。通过多年的努力,公司在某些关键技术领域取得了重要进展,如高速数据传输、低损耗材料应用等。这些技术突破不仅提升了公司的产品性能和质量,也使其在行业中保持了领先地位。

Analytic Instruments Corp公司的发展小趣事

Analytic Instruments Corp成立于XXXX年,最初由几位热衷于电子测量与分析技术的工程师创立。他们凭借对行业的深刻理解和敏锐的市场洞察力,研发出了一款具有创新性的电子测量仪器,该仪器在精度和稳定性上均优于市场上的同类产品。这一技术突破使得公司在创业初期就获得了市场的认可,为后续的发展奠定了坚实的基础。

Goldstar Electron Co Ltd公司的发展小趣事

随着技术的不断成熟和产品的不断完善,Antelec公司开始积极寻求市场拓展。公司通过与国内外知名企业和研究机构建立合作关系,将产品成功应用于多个领域,如通信、半导体制造和航空航天等。同时,公司还积极参加各类行业展会和论坛,加强了与行业内外的交流与合作。

问答坊 | AI 解惑

中国汽车电子:锻造核心竞争力2

天津一汽夏利产品开发中心立足国情 自主创新  “产品开发中心对汽车电子的研究非常重视,因为我们深知只有占领电子技术制高点,才能在市场中赢得主动。”天津一汽夏利产品开发中心(简称产品开发中心)程主任开宗明义强调开发的重要性。  据程主 ...…

查看全部问答>

MCF52259_FEC

MCF52259_FEC—Fast Ethernet Controller   好久没写总结了,游戏一天现在也没什么状态干活,就来总结下近来的成果吧。  调Coldfire的以太网花的时间比较长,以前一直以为这块很难,有种畏惧心理,现在回过头来看看,要是不加协议的话跟IIC、SP ...…

查看全部问答>

wince下过多得new和delete会产生碎片或者内存泄露吗?

在wince系统下,我自己编写了一个链表类,为了存储从串口接收得大量数据,这些数据处理之后就删除了, 长期这样之后会造成大量得碎片吗?或者说是内存泄露吗? 如果出现这样得问题,该怎么处理呢 谢谢…

查看全部问答>

USB摄像头 OV511+OV7620,采集图像不清晰问题

环境:WINCE ARM11 设备:网眼V2000, OV511+USB摄像头 资源:参考网上的WINCE_OV511,带驱动+上位机+PPT讲解的那个 测试结果:YUV420->RGB图像如下。 请了解或者做过的朋友指点下,为什么图像不清晰,而且是花的。谢谢! …

查看全部问答>

socket编程 connect总返回0

在Wince做的Socket程序,客户端首先进行GPRS拨号,成功后在线程中完成connect,但是connect的地址无论什么都返回0,请教一下 为什么? 这个客户端程序移植到VC下则测试 能用。 …

查看全部问答>

STM32定时器若干疑问:)

1.既然:定时器时钟频率是其所在APB 总线频率的两倍。然而,如果相应的APB 预分频系数是1,定时器的时钟频率与所在APB 总线频率一致,岂不是:/* PCLK1 = HCLK/2 */  & ...…

查看全部问答>

请问本版怎么没有版主呢,?

请问本版怎么没有版主呢,? 想交流一下,找不来么,呵呵,我可以么。…

查看全部问答>

面对职场压力:走好“十步棋”

  当你面对职场压力时,学会下面“十步棋”,一定可以变压力为动力,消压力于无形。   第一步,精神超越――价值观和人生定位自我的人生价值和角色定位、人生主要目标的设定等等,简单的说就是:你准备做一个什么样的人,你的人生准备达成哪 ...…

查看全部问答>

怎么获取红外码库?

    请问下要怎么获取红外编码库?我这边是做智能家居红外转WIFI的,面对市场上上百种遥控器,不可能每个按键都学习的方式吧?看了带红外功能的手机里面要100多种红外设备编码库,那么这些编码库要到哪获取?好像网络也找不到这么多。 …

查看全部问答>

献给初学TI 430的童靴们---- MSP430的时钟周期(振荡周期)、机器周期、指令周期

MSP430的时钟周期(振荡周期)、机器周期、指令周期之间的关系 通用知识 时钟周期也称为振荡周期:定义为时钟脉冲的倒数(时钟周期就是直接供内部CPU使用的晶振的倒数,例如12M的晶振,它的时钟周期就是1/12us),是计算机中的最基本的、最小的时 ...…

查看全部问答>