现在的位置: 首页 > 综合 > 正文

(volatile unsigned int *)的理解

2014年05月14日 ⁄ 综合 ⁄ 共 1598字 ⁄ 字号 评论关闭

 

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语言里定义指针并且赋值是这样的:

  1. /*普通的指针赋值*/   |  /*volatile的使用*/  
  2. int *p=NULL;       |  int *p = NULL;   
  3. p=&a;              |  p = (volatile int *)0xa00 //0xa00表示地址  

 

因为c语言中不能直接使用地址,必须转换为指针所以,可以这样理解(volatile int *)0xa00就是一个地址

因为&a实际表示的也是地址,如果我们我们假设储存变量a的地址就是0xa00,那么&a=(volatile int *)0xa00 ;

这样就不难理解了,同时我们上面还可以对上面进一步改造:

  1. //原来的情况  
  2. int *p=NULL;  
  3. p = (volatile int *)0xa00;  
  4.    
  5. //-----------------  
  6.      |  
  7.      |  
  8.      *  
  9. //转换一次  
  10. int q,*p=NULL;  
  11. p = (volatile int *)0xa00;  
  12. q = *p; /*把这个地址存储的值赋值给q*/  
  13. //我们就此可以直接对q赋值相当于对0xa00这个地址赋值  
  14. //-----------------  
  15.      |  
  16.      |  
  17.      *  
  18. //再转换一次  
  19. #define q (*(volatile int *)0xa00)  
  20. 从而就可以直接对q操作了  

 

#define q (*(volatile int *)0xa00)

                 ----------------------------

                          |

                          |________= *p

                                                _

                                                |

                                                |_____=(volatile int *)0xa00 == &a

不知道这样写算不算明白了,参考了http://51growingup.blog.sohu.com/111642657.html里面的内容。

抱歉!评论已关闭.