原创 嵌入式C语言中const与volatile的用法详解

2009-9-29 22:02 3701 6 7 分类: 软件与OS

constvolatile关键字的用法<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />


const最主要的特点就是只读,有常量、常量指针,如果不是特别小心的分析C语言语句的书写格式,再加上指针的使用,就特别容易弄错。


volatile关键字是一个类型修饰符,用它声明的类型变量表示可以被某些编译器未知的因素更改,比如:操作系统、硬件或者其它线程等。遇到这个关键字声明的变量,编译器对访问该变量的代码就不再进行优化,从而可以提供对特殊地址的稳定访问;如果不使用valatile,则编译器将对所声明的语句进行优化。之所以优化是因为访问寄存器要比访问内存单元快得多。但是优化之后容易出现问题,例如现在要直接对内存地址单元的内容修改,如果继续使用未经过valatile声明的变量,则读到的值有可能是寄存器中未经过修改的值,但本意是要读发生变化后的数值,所以会出现意想不到的错误。而经valatile声明的变量,每次访问该变量时都会从内存单元中重新读取。


const经常用于声明不希望被其它程序修改的常量;volatile经常用于声明因意外而可能发生改变的变量。


    下面具体分析两个变量的用法:


1const


关键字const有什么含意?


我只要一听到被面试者说:“const意味着常数,我就知道我正在和一个业余者打交道。其实只要能说出const意味着只读就可以了。尽管这个答案不是完全的答案,但我接受它作为一个正确的答案。如果应试者能正确回答这个问题,我将问他一个附加的问题:


下面的声明都是什么意思?


const int a;


int const a;


const int *a;


int * const a;


int const * a const;


前两个的作用是一样,a是一个常整型数;第三个意味着a是一个指向常整型数的指针(也就是,整型数是不可修改的,但指针可以);第四个意思a是一个指向整型数的常指针(也就是说,指针指向的整型数是可以修改的,但指针是不可修改的);最后一个意味着a是一个指向常整型数的常指针(也就是说,指针指向的整型数是不可修改的,同时指针也是不可修改的)。如果应试者能正确回答这些问题,那么他就给我留下了一个好印象。


2volatile


关键字volatile有什么含意?并给出三个不同的例子。


一个定义为volatile的变量是说该变量可能会被意想不到地改变,这样,编译器就不会去假设该变量的值了。精确地说,优化器在用到该变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。下面是volatile变量的几个例子:


(1) 并行设备的硬件寄存器(如:状态寄存器)


(2) 一个中断服务子程序中会访问到的非自动变量


(2) 多线程应用中被几个任务共享的变量


回答不出这个问题的人是不会被雇佣的。我认为这是区分C程序员和嵌入式系统程序员的最基本的问题。搞嵌入式的同志们经常同硬件、中断、RTOS等等打交道,所有这些都要求用到volatile变量。不懂得volatile的内容将会带来灾难。


假设被面试者正确地回答了这是问题(嗯,怀疑是否会是这样),我将稍微深究一下,看一下这家伙是不是直正懂得volatile完全的重要性。


(1) 一个参数既可以是const还可以是volatile吗?解释为什么。


(2) 一个指针可以是volatile 吗?解释为什么。


(3) 下面的函数有什么错误:


int square(volatile int *ptr)


{


return *ptr * *ptr;


}


下面是答案:


(1) 是的。一个例子是只读的状态寄存器。它是volatile因为它可能被意想不到地改变。它是const因为程序不应该试图去修改它。


(2) 是的。尽管这并不很常见。一个例子是当一个中服务子程序修该一个指向一个buffer的指针时。


(3) 这段代码有点变态。这段代码的目的是用来返回指针*ptr指向值的平方,但是,由于*ptr指向一个volatile型参数,编译器将产生类似下面的代码:


int square(volatile int *ptr)


{


int a,b;


a = *ptr;


b = *ptr;


return a * b;


}


由于*ptr的值可能被意想不到地改变,因此ab可能是不同的。结果,这段代码可能返不是你所期望的平方值!正确的代码如下:


long square(volatile int *ptr)


{


int a;


a = *ptr;


return a * a;


}


 


补充:什么是自动变量


       自动变量就是指在函数内部定义使用的变量。它只允许在定义它的函数内部使用它。在函数外的其他任何地方都不能使用该变量。自动变量是局部变量,即它的区域性是在定义它的函数内部有效。当然这说明自动变量也没有链接性,因为它也不允许其它的文件访问它。由于自动变量在定义它的函数的外面的任何地方都是不可见的,所以允许我们在这个函数外的其它地方或者是其它的函数内部定义同名的变量,它们之间不会发生冲突的。因为它们都有自己的区域性,而其没有链接性(即:不允许其它的文件访问它)

文章评论1条评论)

登录后参与讨论

jiadongchina_618047972 2013-6-20 10:25

“如果不使用valatile,则编译器将对所声明的语句进行优化。”请问此处的优化是什么意思,如何优化
相关推荐阅读
gujunyi1_407560534 2011-03-22 16:43
NP0,X5R,X7R,Y5V释义及其用途
这个是按美国电工协会(EIA)标准,根据不同介质材料而分类。这类参数描述了电容采用的电介质材料类别,温度特性以及误差等参数,不同的值也对应着一定的电容容量的范围。具体来说,就是:X7R常用于容量为33...
gujunyi1_407560534 2011-03-22 16:29
转贴:电容的选型
从电子产品的设计到采购来看,无源元件只是一个配角,其在整个电子产品中所占的物料成本仅约一成,但电子产品中的无源元件数量却是最多的。以电容为例,从2G发展到3G,仅一部手机中的MLCC数量就从平均165...
gujunyi1_407560534 2011-01-28 12:34
手把手将你用PADS Layout 2007.2生成Gerber文件
        本文主要讲述如何利用PADS 2007.2 Layout生成Gerber文件,由于内容较多,所以不再一一列出,这里仅贴出里面第一页的内容,总共14页。需要的博友们可以下载后观看,如有问...
gujunyi1_407560534 2011-01-12 16:10
DxDesigner的两个使用技巧-你一定用得着,而且很有帮助
...
gujunyi1_407560534 2010-11-22 13:47
VCC、VDD、VEE、VSS释疑汇总
在电子电路中,常可以看到VCC、VDD和VSS三种不同的符号,它们有什么区别呢?1、解释VCC:C=Circuit 表示电路的意思,即接入电路的电压;VDD:D=Device 表示器件的意思,即器件内...
gujunyi1_407560534 2010-11-17 12:23
晶体振荡器专题介绍
1、晶振的作用       晶振是晶体振荡器的简称,分为有源晶振和无源晶振两种,有源晶振无需外接匹配电容,只要加电即可输出一定频率的周期波形,所以有源晶振一般是四个引脚;无源晶振严格来说不能叫晶振,只...
我要评论
1
6
关闭 站长推荐上一条 /2 下一条