单片机volatile关键字的作用?

2023-05-16

最近我们无际单片机学员在做stm8和stm32项目的时候,问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.变量读单片机寄存器值

最后彩蛋时间!!!

最近很多小伙伴找我要一些单片机学习资料,然后我根据自己从业十年经验,熬夜肝了几个通宵,精心整理一份「单片机入门到高级教程+工具包」全部无偿共享给大家!!!

除此以外,再含泪分享我压箱底的22个热门开源项目,包含源码+原理图+PCB+说明文档,让你迅速进阶成高手

据说有小伙伴靠这份资料顺利进入BAT大厂,所以保存了一定要好好学习。

教程资料包和详细的学习路径可以看我下面这篇文章的开头。

单片机入门到高级开挂学习路径(附教程+工具)

 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

单片机volatile关键字的作用? 的相关文章

随机推荐

  • 学stm32有机会进大公司吗?如何选择?

    进大公司似乎跟你会什么单片机并没有直接的关系 但凡做过几年研发的都知道 xff0c 虽然我们是从事嵌入式单片机开发 xff0c 但核心其实还是你的编程思维和水平 企业做产品不是固定用哪个单片机的 xff0c 而是根据市场的情况走 xff0c
  • 电气工程及其自动化就业方向及前景

    电气毕业 xff0c 工作10年 xff0c 聊聊这几年的感悟 全文约1800字 xff0c 认真看完 xff0c 相信还在迷茫的你一定茅塞顿开 首先声明一下 xff0c 我实习期间就自学转行进入嵌入式单片机开发了 自己并没有从事过电气工作
  • 大学如何自学嵌入式开发?

    我2011年就开始从事嵌入式软件开发 很多人觉得学习嵌入式开发比高考还难 xff0c 不知道怎么入门 xff1f 要学哪些东西 xff1f 嵌入式开发涉及的知识 xff0c 你确实一辈子都学不完 但是 xff0c 这不代表你需要花费很长的时
  • 嵌入式工作会越来越少吗?

    学习技术并非易事 xff0c 就像唐僧取西经 xff0c 需要经历皮肉之苦和精神上的折磨 学习最迷茫的无非是不确定自己能否学会 xff0c 学会以后能否落地 xff0c 给自己事业赋能 这种焦虑伴随着我当初整个自学过程 xff0c 直到找到
  • 解决ubuntu linux触摸板不能用问题

    解决ubuntu linux触摸板不能用问题 2011 04 16 01 40 其实很简单 xff1a 打开终端 xff0c 然后sudo i获取root权限 xff0c 然后 xff1a sudo modprobe r psmouse s
  • 一文搞懂栈(stack)、堆(heap)、单片机裸机内存管理malloc

    大家好 xff0c 我是无际 有一周没水文了 xff0c 俗话说夜路走多了难免遇到鬼 最近就被一个热心网友喷了 说我的文章没啥营养 xff0c 所以今天来一篇烧脑的 哈哈 xff0c 开个玩笑 xff0c 不要脸就没人能把我绑架 主要是最近
  • spi协议时序图和四种模式实际应用详解

    大家好 xff0c 我是无际 上个章节我们讲解了spi接口定义 xff0c 今天我们更加深入讲解下spi协议时序图和spi四种模式的用法 刚开始接触单片机开发时 xff0c 最怕就是看时序图 xff0c 对于我来说就是奇怪的知识 特别是SP
  • 你用esp8266做出过什么有趣的东西?

    曾经有个40多岁的老板跟我说 xff0c 很羡慕我这种做技术的 xff0c 他也喜欢钻研 xff0c 感觉自己做产品很有成就感 男人是天生的小朋友 xff0c 都喜欢玩 xff0c 都喜欢钻研 我做单片机开发11年了 xff0c 深感技术的
  • stm32正点原子和普中或是野火哪个好?

    看了很多热心网友对正点原子 野火 普中等开发板的对比和评论 说这个讲的不行 xff0c 那个代码写得垃圾 xff0c 还有谁谁谁的像一坨屎 说真的 xff0c 有种卸磨杀驴的感觉 为什么这么说 xff1f 因为忽略了两个问题 xff1a 1
  • 嵌入式Linux和stm32区别? 之间有什么关系吗?

    本人从事嵌入式单片机开发11年 xff0c 很多人对嵌入式 单片机 STM32 Linux傻傻分不清楚 导致学习过程走了很多弯路 xff0c 明明只需要几个月学习时间 xff0c 却浪费了1 2年 xff0c 最后抱怨嵌入式也太高了吧 xf
  • 为什么有些人说单片机简单,我学起来这么吃力?

    相信我 xff0c 你不是学不会 xff0c 只是没找对方法 看完这篇文章 xff0c 你就发现原来自己学起来并没那么吃力 开始前点赞 43 收藏赶紧点起来 xff0c 不然我怕你看得太入神 xff0c 下次想看都找不到 很多人学单片机从入
  • 嵌入式系统中,FLASH中的程序代码必须搬到RAM中运行吗?

    这个问题 xff0c 核心还是得看你用什么芯片方案 xff01 嵌入式系统涉及技术栈特别庞大 xff0c 不仅仅大家理解的ARM 43 Linux而已 我从业11年 xff0c 也只敢说对其中一个方向熟悉而已 嵌入式系统中 xff0c 核心
  • stm32入门开发板选野火还是正点原子呢?

    两个可以说都是行业标杆 不能说哪个比哪个更好 xff0c 各有特色 xff0c 适合自己就是最好的 xff0c 同行之间都是相互参考去做的 我从以下几个维度来对比下 xff1a 1 教程 2 代码 3 开发板价格 4 进阶 一 教程 两家教
  • 树莓派为什么涨价了?

    任何东西涨价都是一个原因 xff1a 供不应求 别说树莓派 xff0c 很多芯片都涨价了 xff0c 比如说STM32 xff0c 去年最高峰涨了将近10倍 所以 xff0c 引发了很多有趣的事 比如说 xff1a 以前接到订单老板开心得睡
  • jtag和swd区别,该用哪个?

    我刚开始学习单片机是STC最经典的STC89C52RC 导致我后面很多51的产品应用都会选择STC xff0c 一方面是因为熟悉 xff0c 一方面是使用确实很方便 其中最让我满意的就是烧录 xff0c 可以直接用串口就可以下载程序 xff
  • 匈牙利算法原理与Python实现

    匈牙利算法原理与Python实现 今天学习一个新的算法 匈牙利算法 xff0c 用于聚类结果分析 xff0c 先用图表示我当前遇到的问题 xff1a 这两列值是我用不同算法得到的聚类结果 xff0c 从肉眼可以看出第一列聚类为0的结果在第二
  • STM32之后哪块板子发展最好?

    从事开发10年了 xff0c 我来说说我个人见解 首先 xff0c 除非是顶尖的垄断技术 xff0c 否则一般不会有屹立不倒的芯片 站在研发经理的角度 xff0c 单片机选型时主要会考虑以下几个因素 xff1a 1 稳定性 2 价格 3 交
  • stm32用什么软件编程烧写比较好呢?

    今天给大家分享我从业单片机开发11年必备的神器 在刚入行不久 xff0c 看到那些老工程师费很大的劲去搭建开发调试环境 xff0c 让我很不解 我想有那工夫 xff0c 我程序都写完了 xff0c 如果有问题就看代码找Bug呗 直到后面做了
  • 嵌入式开发要学多久?要学哪些课程

    曾经有新手和我抱怨 xff1a 为什么嵌入式开发这么难 xff1f 相信我 xff0c 你不是学不会 xff0c 只是没找对方法 这是一片真正为 终结迷茫 而生的回答 xff0c 内容有点长 xff0c 如果你赶时间 xff0c 建议先点赞
  • 单片机volatile关键字的作用?

    最近我们无际单片机学员在做stm8和stm32项目的时候 xff0c 问volatile这个关键词怎么理解 xff1f 有什么作用 xff1f Volatile是C语言的一个关键字 xff0c 在stm8和stm32的固件库里也会经常看见这