volatile常用于多线程共享资源和嵌入式软件的中断。
(一)嵌入式软件中断中volatile的应用
volatile unsigned short g_timer3_count = 0;void TIM3_IRQHandler(void)
{if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET){TIM_ClearITPendingBit(TIM3, TIM_IT_Update);g_timer3_count++;}
}
TIM3_IRQHandler是定时器3的中断服务函数。定时器3配置为每1ms进入一次中断,每次进入全局变量g_timer3_count自加1。
在另一个.c里调用g_timer3_count。
extern volatile unsigned short g_timer3_count;void func(void)
{if(3000 == g_timer3_count){//do something...}
}
如果不使用volatile修饰g_timer3_count的话,很有可能永远进入不了if(3000 == g_timer3_count)的判断里,因为判断里的g_timer3_count的值可能为固定的某个值。
造成这种现象的原因有两种:
1、编译器优化掉了。编译器认为这个变量不会有变化,但实际上在硬件或者多线程中已经改变了。所以要加volatile来告诉编译器这个变量不需要优化,变量的值每次直接从内存读取;
2、变量在其他地方改变,而cash里的值没有改变,每次从cash里面获取。因此也要加volatile,每次不用cash而直接从内存读取。
volatile关键字表示每次读写该内存单元中的数据时都要到内存单元处去取,而不是读取寄存器中的备份值。