嵌入式开发-c语言中volatile关键字作用
C 语言语言中中volatile 关键字关键字作用作用
语言语言中中 关键字关键字作用作用
一个定义为volatile 的变量是说这变量可能会被意想不到地改变,这样,编
译器就不会去假设这个变量的值了。精确地说就是,优化器在用到这个变量时必
须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。下
面是volatile 变量的几个例子:
1). 并行设备的硬件寄存器 (如:状态寄存器)
2). 一个中断服务子程序中会访问到的非自动变量(Non-automatic vari
ables)
3). 多线程应用中被几个任务共享的变量
回答不出这个问题的人是不会被雇佣的。我认为这是区分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)
{
15
int a,b;
a = *ptr;
b = *ptr;
return a * b;
}
由于*ptr 的值可能被意想不到地该变,因此a 和b 可能是不同的。结果,
这段代码可能返不是你所期望的平方值!正确的代码如下:
long square(volatile int *ptr)
{
int a;
a = *ptr;
return a * a;
}
Volatile 关键字告诉编译器不要持有变量的临时性拷贝。一般用在多线程程序
中,以避免在其中一个线程操作该变量时,将其拷贝入寄存器。请看以下情形:
A 线程将变量复制入寄存器,然后进入循环,反复检测寄存器的值是否满足
一定条件(它期待B 线程改变变量的值。
在此种情况下,当 B 线程改变了变量的值时,已改变的值对其在寄存器的值没
有影响。所以A 线程进入死循环。
volatile 就是在此种情况下使用。
16