许多程序员无法正确的理解C语言关键字volatile。这并不奇怪,大多数C原因书籍不过一两句一带而过。本文将告诉你如何正确使用它。

在C/C++嵌入式代码中,你是否经历过下面的情况:

● 代码执行正常–直到你打开了编译器优化

● 代码执行正常–直到打开了中断

● 古怪的硬件驱动

● RTOS的任务独立运行正常–直到生成了其他任务

如果你的回答是“yes”,很有可能你没有使用C语言关键字volatile。你并不是唯一的,很多程序员都不能正确使用volatile。不幸的是,大多数c语言书籍对volatile的藐视,只是简单地一带而过。

volatile是用于声明变量时的使用限定符。它告诉编译器该变量值可能随时发生变化,且这种变化并不是代码引起的。给编译器这个暗示是很重要的。在开始前,我们向来看一看volatile的语法。

C语言关键字volatile语法

声明一个变量为volatile,可以在数据类型之前或之后加上关键字volatile。下面的语句,把foo声明一个volatile的整型。

volatile int foo;

int volatile foo;

把指针指向的变量声明为volatile很常见,尤其是I/O寄存器的地址映射。下面的语句,把pReg声明为一个指向8-bit无符号指针,指针指向的内容为volatile。

volatile uint8_t * pReg;

uint8_t volatile * pReg;

volatile的指针指向非volatile的变量很少见(我只使用过一次),但我还是给出相应的语法。

int * volatile p;

顺便提一下,关于为什么要在数据类型前使用volatile关键字,请自行百度搜素。

最后,如果你在struct或者union前使用volatile关键字,表明struct或者union的所有内容都是volatile。如果这不是你的本意,可以在struct或者union成员上使用volatile关键字。

正确使用C语言关键字volatile

只要变量可能被意外的修改,就需要把该变量声明为volatile。实际应用中,只有三种类型数据可能被修改。

  1. 外设寄存器地址映射

  2. 在中断服务程序中修改全局变量

  3. 在多线程、多任务应用中,全局变量被多个任务读写

我们将分别讨论上述三种情况。

外设寄存器

嵌入式系统包含真正的硬件,通常会有复杂的外设。这些外设寄存器的值可能被异步的修改。举个简单的例子,我们要把一个8-bit状态寄存器的地址映射到0x1234.在程序中循环查看该状态寄存器的值是否变为非0. 下面是最容易想到,但错误的实现方法

当你打开编译器优化时,程序总是执行失败。因为编译器会生成下面的汇编代码:

程序被优化的原因很简单,既然已经把变量的值读入累加器,就没有必要重新写一遍,编译器认为值是不会变化的。就这样,在第三行,程序进入了无限死循环。为了告诉编译器我们的真正意图,我们需要修改函数的声明:

编译器生成的汇编代码:

像这样,我们得到了正确的动作。

中断服务程序

在中断服务程序中,经常会修改一些全局变量值,来作为主程序中的判断条件。例如,在串口中断服务程序中,可能会检测是否接收到了ETX(假如是消息的结束标识符)字符。如果接收到了ETX,ISR设置一个全局标志位。

错误的做法:

在关闭编译器优化的情况下,程序可能执行正常。然而,任何像样点而优化都会“break”这段程序。问题是编译器并不知道etx_rcvd可能被ISR中被修改。编译器只知道,表达式!ext_rcvd始终为真,你将永远无法退出循环。结果,循环后面的代码可能被编译器优化掉。幸运的话,你的编译器可能会发出警告;不幸的话,(或者你不认真的查看编译器警告),你的程序无法正常执行。当然,你可以责怪编译器执行了“糟糕的优化”。

解决方式是,将变量etx_rcvd声明为volatile,所有问题(当然,也可能是部分)就消失了。

多线程应用

在实时系统中,尽管有像queues,pipes等这些同步机制,使用全局变量实现两个任务共享信息的做法依然很常见。即使在你的程序中加入了抢占式调度器,你的编译器依然无法知道什么是上下文切换,或何时发生上下文切换。因此,从概念上讲,多任务修改全局变量的的做法与中断服务程序中修改全局变量的做法是相同的。因此,所有这类全局变量都应该声明为volatile。例如,下面的程序

当打开编译器优化时,这段程序可能执行失败。解决方法是将cntr声明为volatile。

最后的思考

一些编译器允许你把所有的变量隐式的声明为volatile。请抵制这种诱惑,因为它会令你不再思考,当然,也会导致生成低效的代码。

另外,也不要责怪优化器或直接把它关掉。现代的优化器已经足够优秀,我已经记不清上次遇到优化bug是什么时候了。相反,我常常看到程序员们错误地使用volatile。

如果你被要求去修改一个很古怪的代码,请在程序中查找一下volatile关键字,如果你什么也没有找到,上面讨论的例子可以向你提供一些解决问题的思路。

本文章来源网络,如果原作者不支持咱们转发,请联系删除,谢谢

volatile用法相关推荐

  1. c中volatile用法

    转载自:http://www.cnblogs.com/chio/archive/2007/11/24/970632.html 感谢原创分享! volatile 影响编译器编译的结果,指出,volati ...

  2. C中的volatile用法

    volatile 影响编译器编译的结果,指出,volatile 变量是随时可能发生变化的,与volatile变量有关的运算,不要进行编译优化,以免出错,(VC++ 在产生release版可执行码时会进 ...

  3. volatile用法详解

    我在 开发 中也常常遇到这个问题,发现通常用在两个方面,一方面 是 对 硬件 寄存器 或固定内存 的 访问,一般要用到,这就是我们常常在寄存器的头 文件 常常看到的,另一个就是在 多 线程,或主 程序 ...

  4. java volatile 用法_java关键字volatile用法详解

    volatile关键字想必大家都不陌生,在java 5之前有着挺大的争议,在java 5之后才逐渐被大家接受,同时作为java的关键字之一,其作用自然是不可小觑的,要知道它是java.util.con ...

  5. C语言之volatile用法(二十一)

    一.volatile 介绍 参看:volatile详解 参看:C Language Keywords Indicates that a variable can be changed by a bac ...

  6. C语言之volatile用法(二十一),2021最新Android面试笔试题目分享

    int main (void) { int i = 10; int a = i; //优化 int b = i; printf ("i = %d\n", b); return 0; ...

  7. 一文彻底搞懂volatile用法

    一.常见说法 volatile 关键字和const对应,一样是一种类型修饰符,用它修饰的变量表示可以被某些编译器未知的因素更改,比如操作系统.硬件或者其它线程等.遇到这个关键字声明的变量,编译器对访问 ...

  8. 多线程并发的解决方案 volatile synchronized notify notifyAll wait关键字分析

    2019独角兽企业重金招聘Python工程师标准>>> 1.ThreadLocal用法 在java中,如果一个变量需要被多个线程访问,可以使用volatile来声明它为"易 ...

  9. volatile关键字对

    今天的工作中,我遇到了Java中的volatile关键字. 不太熟悉,我发现了以下解释: Java理论与实践:管理波动 鉴于该文章详细解释了所讨论的关键字,您是否曾经使用过它,或者是否曾见过可以正确使 ...

最新文章

  1. DotNetNuke CSS hierarchy
  2. IBatis存取图片在Oracle Blob大字段中Asp.Net
  3. Photo Stack效果
  4. layui 鼠标移入变为小手_游戏技术上不去?看看外设选对没!鼠标篇
  5. 前端开源项目周报0307
  6. Modbus协议栈开发笔记之二:Modbus消息帧的生成
  7. Unity中uGUI的控件事件穿透逻辑
  8. Java基础学习(4)-动态代理
  9. ios不能保存png_ios转一加8t使用体验随用随更新
  10. TK1装kuboki的USB驱动和TK1的无线网卡驱动
  11. 关于写的Java书籍进展
  12. 计算机开机密码输入不了,win10开机密码输入不了,win10开机密码输入没反应
  13. 常用软件分类 精选列表(二)
  14. 小米C++开发笔试真题
  15. 如何用计算机作图并求斜率,简单斜率分析以及作图
  16. 2018年应届毕业宇视科技嵌入式软件开发工程师面试笔试总结
  17. 学历对做软件测试的影响
  18. 澳洲网:超8成雇主对澳大利亚高校毕业生表示满意
  19. C#ListView操作一二
  20. VirtualBox虚拟机网络设置(四种方式)

热门文章

  1. 书籍推荐-记这几年看的书
  2. 输入流与输出流的区别
  3. html 缩略图点击预览,[每天进步一点点~] uni-app 点击图片实现预览图片列表
  4. HTML段落,换行,字符实体
  5. 【数据分析】reshape(-1,1)和numpy的广播机制
  6. oracle sql 语句 start with ...... connect by prior .......
  7. 图像处理之积分图应用三(基于NCC快速相似度匹配算法)
  8. grpc php 返回值过大,使用grpc实现php、java、go三方互调
  9. 第七季1:MP4文件格式解析
  10. 5.spiders(文件夹)