C/C++ 中的 volatile 关键字和 const 对应,用来修饰变量,通常用于建立语言级别的 memory barrier。这是 BS 在 "The C++ Programming Language" 对 volatile 修饰词的说明:
A volatile specifier is a hint to a compiler that an object may change its value in ways not specified by the language so that aggressive optimizations must be avoided.
volatile 指出 i 是随时可能发生变化的,每次使用它的时候必须从 i的地址中读取,因而编译器生成的汇编代码会重新从i的地址读取数据放在 b 中。而优化做法是,由于编译器发现两次从 i读数据的代码之间的代码没有对 i 进行过操作,它会自动把上次读的数据放在 b 中。而不是重新从 i 里面读。这样以来,如果 i是一个寄存器变量或者表示一个端口数据就容易出错,所以说 volatile 可以保证对特殊地址的稳定访问。注意,在 VC 6 中,一般调试模式没有进行代码优化,所以这个关键字的作用看不出来。下面通过插入汇编代码,测试有无 volatile 关键字,对程序最终代码的影响:
输入下面的代码:
01
#include <stdio.h>
02
03
voidmain()
04
{
05
inti = 10;
06
inta = i;
07
08
printf("i = %d", a);
09
10
// 下面汇编语句的作用就是改变内存中 i 的值
11
// 但是又不让编译器知道
12
__asm {
13
mov dword ptr [ebp-4], 20h
14
}
15
16
intb = i;
17
printf("i = %d", b);
18
}
然后,在 Debug 版本模式运行程序,输出结果如下:
i = 10 i = 32
然后,在 Release 版本模式运行程序,输出结果如下:
i = 10 i = 10
输出的结果明显表明,Release 模式下,编译器对代码进行了优化,第二次没有输出正确的 i 值。下面,我们把 i 的声明加上 volatile 关键字,看看有什么变化: