关于volatile的用法:
1:它的作用是告诉编译器volatile变量是随时可能发生变化的,与volatile变量有关的运算,不要自作主张进行编译优化,以免出错,例如:
volatile int i=10;
int j = i;
int k = i;
……
volatile 告诉编译器i是随时可能发生变化的,每次使用它的时候必须从i的地址中读取,因而编译器生成的可执行码会重新从i的地址读取数据放在k中。
而优化做法是,由于编译器发现两次从i读数据的代码之间的代码没有对i进行过操作,它会自动把上次读的数据放在k中。而不是重新从i里面读。这样以来,如果i是一个寄存器变量或者表示一个端口数据就容易出错,所以说volatile可以保证对特殊地址的稳定访问,不会出错。
2:第二点是强制指针转换。我们平常编程序的时候都是直接对一些寄存器或端口直接操作,比如51单片机的P0(1,2,3),但是为什么对P0口操作就是对地址80H进行操作呢,在C51中是由关键字sfr来定义的,但在其他单片机中没有这个关键字,就要用强制指针来定义了,(unsigned int *)不管后面是什么,都把它强制转换为unsigned int类型,*(unsigned int *) 0x8001表示取这个指针的内容,cnt = *(unsigned int *) 0x8001把这个内容赋值给cnt,以后就可以直接对cnt进行操作了,也就相当于对地址0x8001进行操作。能不能对0x8001进行直接操作?cnt=0x8001或cnt=8001,错误,都达不到你的目的,所以必须要用指针。
进一步理解:
C语言里定义指针并且赋值是这样的:
- /*普通的指针赋值*/ | /*volatile的使用*/
- int *p=NULL; | int *p = NULL;
- p=&a; | p = (volatile int *)0xa00 //0xa00表示地址
因为c语言中不能直接使用地址,必须转换为指针所以,可以这样理解(volatile int *)0xa00就是一个地址
因为&a实际表示的也是地址,如果我们我们假设储存变量a的地址就是0xa00,那么&a=(volatile int *)0xa00 ;
这样就不难理解了,同时我们上面还可以对上面进一步改造:
- //原来的情况
- int *p=NULL;
- p = (volatile int *)0xa00;
- //-----------------
- |
- |
- *
- //转换一次
- int q,*p=NULL;
- p = (volatile int *)0xa00;
- q = *p; /*把这个地址存储的值赋值给q*/
- //我们就此可以直接对q赋值相当于对0xa00这个地址赋值
- //-----------------
- |
- |
- *
- //再转换一次
- #define q (*(volatile int *)0xa00)
- 从而就可以直接对q操作了
#define q (*(volatile int *)0xa00)
----------------------------
|
|________= *p
_
|
|_____=(volatile int *)0xa00 == &a
不知道这样写算不算明白了,参考了http://51growingup.blog.sohu.com/111642657.html里面的内容。