volatile要害字的说明以及测试
volatile要害字是一种范例修饰符,用它声明的范例变量暗示可以被某些编译器未知的因素变动,好比:操纵系统、硬件可能其它线程等。碰着这个要害字声明的变量,编译器对会见该变量的代码就不再举办优化,从而可以提供对非凡地点的不变会见。
利用该要害字的例子如下:
int volatile nVint;
当要求利用volatile 声明的变量的值的时候,系统老是从头从它地址的内存读取数据,纵然它前面的指令方才从该处读取过数据。并且读取的数据立即被生存。
譬喻:
volatile int i=10;
int a = i;
...
//其他代码,并未明晰汇报编译器,对i举办过操纵
int b = i;
volatile 指出 i是随时大概产生变革的,每次利用它的时候必需从i的地点中读取,因而编译器生成的汇编代码会从头从i的地点读取数据放在b中。而优化做法是,由于编译器发明两次从i读数据的代码之间的代码没有对i举办过操纵,它会自动把上次读的数据放在b中。而不是从头从i内里读。这样以来,假如i是一个寄存器变量可能暗示一个端口数据就容易堕落,所以说volatile可以担保对非凡地点的不变会见。
留意,在vc6中,一般调试模式没有举办代码优化,所以这个要害字的浸染看不出来。下面通过插入汇编代码,测试有无volatile要害字,对措施最终代码的影响:
首先,用classwizard建一个win32 console工程,插入一个voltest.cpp文件,输入下面的代码:
#include <stdio.h>
void main()
{
int i=10;
int a = i;
printf("i= %d\n",a);
//下面汇编语句的浸染就是改变内存中i的值,可是又不让编译器知道
__asm {
mov dword ptr [ebp-4], 20h
}
int b = i;
printf("i= %d\n",b);
}
然后,在调试版本模式运行措施,输出功效如下:
i = 10
i = 32
然后,在release版本模式运行措施,输出功效如下:
i = 10
i = 10
输出的功效明明表白,release模式下,编译器对代码举办了优化,第二次没有输出正确的i值。下面,我们把 i的声明加上volatile要害字,看看有什么变革:
#include <stdio.h>
void main()
{
volatile int i=10;
int a = i;
printf("i= %d\n",a);
__asm {
mov dword ptr [ebp-4], 20h
}
int b = i;
printf("i= %d\n",b);
}
别离在调试版本和release版本运行措施,输出都是:
i = 10
i = 32
这说明这个要害字发挥了它的浸染!