原文

概要

许多程序员无法正确的理解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关键字,请参考Dan Sak的文章Top-Level cv-Qualifiers in Function Parameters

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

正确使用C语言关键字volatile

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

  1. 外设寄存器地址映射
  2. 在中断服务程序中修改全局变量
  3. 在多线程、多任务应用中,全局变量被多个任务读写

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

外设寄存器

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

uint8_t * pReg = (uint8_t *)0x1234;//wait for register to become non-zero
while(*pReg == 0){} //do something else

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

mov ptr, #0x1234
mov a, @ptrloop:
bz loop

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

uinit8_t volatile * pReg = (uint8_t volatile *)0x1234;

编译器生成的汇编代码:

mov ptr, #0x1234loop:
mov a, @ptr
bz loop

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

中断服务程序

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

int etx_rcvd = FALSE;
void main()
{...while(!ext_rcvd){//wait}...
}interrupt void rx_isr(void)
{...if(ETX == rx_char){etx_rcvd = TRUE;}...
}

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

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

多线程应用

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

int cntr;void task1(void)
{cntr = 0;while(cntr == 0){sleep(1);}...
}void task2(void)
{...cntr++sleep(10);...
}

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

最后的思考

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

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

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

关于作者

Nigel Jones,著名嵌入式专家,有很多经典文章,如《A ‘C’ Test: The 0x10 Best Questions for Would-be Embedded Programmers》,中文《想成为嵌入式程序员应知道的0x10个基本问题》;《Efficient C Code for 8-bit Microcontrollers》。

Nigel Jones 博客地址

参考

  1. 如何使用C语言中的volatile关键字
  2. How to Use C’s volatile keyword
  3. Volatile陷阱
  4. 详解C中volatile关键字
  5. volatile关键字
  6. C语言中volatile关键字的作用

【译】volatile C语言关键字,如何使用?相关推荐

  1. 从关键字~C语言 — 期末考,考研,面试中那些你不得不知道的C语言关键字细节

    目录 引言: 初步认识了C语言的结构之后 博主将带您进入关键字的世界了 1. 关于变量 1.1 什么是变量 1.2 变量的定义与声明 1.3 为什么要定义变量 1.4 变量定义的本质 2. 关键字 2 ...

  2. c语言 char转int_C语言关键字及进制的转换你都知道吗?

    ​前面我们讲过 C语言简洁.紧凑 使用方便.灵活 那是什么使得C语言这么方便呢? 那就是关键字,或称保留字 C语言的关键字共有32个 根据关键字的作用 可分为 数据类型关键字 控制语句关键字 存储类型 ...

  3. c语言字母表关键字,读书笔记-C语言关键字

    001 关键字 C语言一共32个关键字 1. 声明和定义 在开始认识关键字前,必须要明白什么是声明,什么事定义: 定义:(编译器)创建一个对象,为这个对象分配一段内存并给他取上一个名字.在一个作用域内 ...

  4. c语言getchar用法_C语言 — 关键字

    几十个关键字不多,用得多了自然会记住,相信大家也不会担心.下面是C语言中的 32 个关键字: 一丶C语言关键字 第一个关键字:auto 用来声明自动变量.可以显式的声明变量为自动变量.只要不是声明在所 ...

  5. java里的关键字有什么用_java语言关键字有哪些?都有什么用处?

    Java语言中的关键字是非常多的,并且有着一定的规范,本文就将带大家深入了解其中详情. 关键字概念 Java中的关键字一般会被用来表示一种数据类型,或表示程序的结构等;它们是不能用作变量名.方法名.类 ...

  6. 根据作用C语言关键字分为,C语言 关键字

    关键字就是已被C语言本身使用,不能作其它用途使用的字.例如关键字不能用作变量名.函数名等 由ANSI标准定义的C语言关键字共32个. 根据关键字的作用,可以将关键字分为数据类型关键字和流程控制关键字两 ...

  7. 由ANSI标准定义的C语言关键字,C语言32个关键字详解

    C语言中32个关键字详解 由ANSI标准定义的C语言关键字共32个: auto double int struct break else long switch case enum register ...

  8. C语言关键字浅析-enum

    ### C语言关键字浅析系列 ### ### ISO/ANSI C 关键字 ### enum关键字的原型是enumerated type,意思是枚举类型 这是ISO/ANSI C90标准新增的关键字之 ...

  9. 什么是C语言的合法正确标识符?什么是C语言关键字?

    C语言标识符与关键字 一.C语言标识符 1.1 标识符的概念 1.2 C语言中合法的标识符有以下这些: 1.3 C语言中非法的标识符有以下这些: 1.4 标识符使用注意事项: 二.C语言关键字 2.1 ...

最新文章

  1. npj Digit. Med. | 基于人工智能的FDA批准的医疗器械和算法现状
  2. 对象序列化与反序列化
  3. [git/svn]Git和SVN差异
  4. 《JAVA与模式》之工厂方法模式
  5. 【Python】集合的交、并、补、差集怎么算?
  6. virtualbox和vagrant卸载脚本在macbook
  7. oracle 的一些基础查询
  8. 外卖和快递行业数据_下周一起,整治全面启动!锁定全市外卖、快递行业!
  9. 对象空指针_可选和对象:空指针救星!
  10. Jakarta EE中的规范范围
  11. 面向对象思想精华总结
  12. 华为还是输了!双11战报出炉,离苹果仍有距离
  13. 干货来啦!!!二十种Python代码游戏源代码分享
  14. python写图片爬取软件_python抓取整个网站图片
  15. Vue 记录一次安装插件引起的项目崩溃(This is probably not a problem with npm,there is likely additional logging outp)
  16. 基于LSTM神经网络的负荷预测(Python代码实现)
  17. 6.2.1邻接矩阵法
  18. 程序员的贫富两极分化,穷的穷死,富的富死,我就是那“穷鬼”?
  19. Revit一款主要用于进行建筑信息建模的软件
  20. Win10计算机窗口空白,Win10系统下启用或关闭windows功能打开后显示空白如何解决...

热门文章

  1. 抗混叠滤波matlab实现,关于设计抗混叠滤波器容易忽视的三条建议
  2. 学习记录:安装binwalk
  3. 一文读懂STM32时钟树(时钟系统)(以STM32F767为例)
  4. VisualSVN的下载与安装
  5. java+OpenCV3 +百度OCR(或tesseract) 识别表格数据
  6. mycat原理及分表分库入门
  7. linux默认端口范围是多少?
  8. ps切片成html形式怎么用,PS教程:Photoshop切片工具把效果图转成网页格式 PS入门综合 - PS学习网...
  9. 如何成为一名合格的DBA
  10. 雨听 | 英语学习笔记(七)~作文范文:学生退学