关于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语言里定义指针并且赋值是这样的:
因为c语言中不能直接使用地址,必须转换为指针所以,可以这样理解(volatile int *)0xa00就是一个地址
因为&a实际表示的也是地址,如果我们我们假设储存变量a的地址就是0xa00,那么&a=(volatile int *)0xa00 ;
这样就不难理解了,同时我们上面还可以对上面进一步改造:
#define q (*(volatile int *)0xa00)
----------------------------
|
|________= *p
_
|
|_____=(volatile int *)0xa00 == &a
不知道这样写算不算明白了,参考了http://51growingup.blog.sohu.com/111642657.html里面的内容。
本文详细介绍了volatile关键字的作用,特别是在编译器优化时如何确保对变量的正确访问。volatile用于告诉编译器相关变量可能随时变化,避免优化导致的错误。此外,还讲解了在没有sfr关键字的单片机环境中,如何使用volatile和强制指针转换来操作特定地址,例如通过(volatile int *)0xa00来访问和赋值。最后,通过示例展示了如何通过多次转换直接对地址进行操作。

3454

被折叠的 条评论
为什么被折叠?



