【002 关键字】一文彻底搞懂volatile用法

一、常见说法

volatile 关键字和const对应,一样是一种类型修饰符,用它修饰的变量表示可以被某些编译器未知的因素更改,比如操作系统、硬件或者其它线程等。遇到这个关键字声明的变量,编译器对访问该变量的代码就不再进行优化,从而可以提供对特殊地址的稳定访问。当要求使用 volatile 声明的变量的值的时候,系统总是重新从它所在的内存读取数据,即使它前面的指令刚刚从该处读取过数据。············


二、volatile作用

 1、 举例说明作用

int i=10;
int j = i;//(1)语句
int k = i;//(2)语句

这时候编译器对代码进行优化,因为在(1)、(2)两条语句中,i 没有被用作左值。这时候编译器认为i 的值没有发生改变,所以在(1)语句时从内存中取出 i 的值赋给 j 之后,这个值并没有被丢掉,而是在(2)语句时继续用这个值给 k 赋值。编译器不会生成出汇编代码重新从内存里取 i 的值,这样提高了效率。但要注意:(1)、(2)语句之间 i 没有被用作左值才行。

volatile int i=10;
int j = i;//(3)语句
int k = i;//(4)语句

volatile 关键字告诉编译器 i 是随时可能发生变化的,每次使用它的时候必须从内存中取出 i 的值,因而编译器生成的汇编代码会重新从 i 的地址处读取数据放在 k 中
 
所以说使用 volatile 声明的变量的值的时候,系统总是重新从它所在的内存读取数据,即使它前面的指令刚刚从该处读取过数据。精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。

2、volatile使用场景

a、中断服务程序中修改的(供其它程序检测的)变量需要加 volatile;

当变量在触发某中断程序中修改,而编译器判断主函数里面没有修改该变量,因此可能只执行一次从内存到某寄存器的读操作,而后每次只会从该寄存器中读取变量副本,使得中断程序的操作被短路(“做与不做一个样”,相当于中断对变量的操作,编译器根本不知道,编译器只是从副本中 获取“实时”的值)。

b、多线程应用中被几个任务共享的变量应该加 volatile;

简单地说就是防止编译器对代码进行优化,比如如下程序:

XBYTE[2]=0x55;
XBYTE[2]=0x56;
XBYTE[2]=0x57;
XBYTE[2]=0x58;

对外部硬件而言,上述四条语句分别表示不同的操作,会产生四种不同的动作,但是编译器却会对上述四条语句进行优化,认为只有XBYTE[2]=0x58(即忽略前三条语句,只产生一条机器码)。 

也就是说,当读取一个变量时,编译器优化有时会先把变量读取到一个寄存器中;以后,再取变量值时,就直接从寄存器中取值;当内存变量或寄存器变量因别的线程等而改变了值,该寄存器的值不会相应改变,从而造成应用程序读取的值和实际的变量值不一致 。

c、并行设备的硬件寄存器(如:状态寄存器) 

RGB屏幕状态寄存器是用于存储RGB屏幕的状态信息和错误码的寄存器,通常会被多个处理单元和中断服务程序访问。在多线程或多任务的环境中,主状态寄存器可能会被多个线程或任务同时访问,因此需要使用volatile修饰符来确保读写操作的原子性和可见性。

存储器映射的硬件寄存器通常也要加voliate,因为每次对它的读写都可能有不同意义。

例如:假设要对一个设备进行初始化,此设备的某一个寄存器为0xff800000。

int *output = (unsigned int *)0xff800000;//定义一个IO端口;
int init(void)
{
    int i;
    for(i=0;i< 10;i++){
    *output = i;
    }
}

经过编译器优化后,编译器认为前面循环半天都是废话,对最后的结果毫无影响,因为最终只是将output这个指针赋值为 9,所以编译器最后给你编译编译的代码结果相当于:

int init(void)
{
    *output =9;
}

如果你对此外部设备进行初始化的过程是必须是像上面代码一样顺序的对其赋值,显然优化过程并不能达到目的编译器在优化后,也许你的代码对此地址的读写操作只做了一次。然而从代码角度看是没有任何问题的。这时候就该使用volatile通知编译器这个变量是一个不稳定的,在遇到此变量时候不要优化。

因此,在imx6ull开发中,对硬件寄存器指针的定义常用到volatile修饰。


三、volatile指针

volatile char* reg;//修饰由指针指向的对象、数据是 volatile 的:

这里定义了一个uchar类型的指针,并且这个指针指向的是一个volatile的对象但是指针变量本身并不是volatile的如果对指针变量reg本身进行计算或者赋值等操作,是可能会被编译器优化的。但是对reg所指向的内容reg的引用却禁止编译器优化。通常在驱动程序的开发中,对硬件寄存器指针的定义,都应该采用这种形式。

uchar * volatile reg;//修饰指针变量本身是volatile的

这里定义了一个uchar类型的指针,并且这个指针变量本身是volatile 的,但是指针所指的内容并不是volatile的!实际使用的时候,编译器对代码中指针变量reg本身的操作不会进行优化,但是对reg所指的内容reg却会作为non-volatile内容处理,也就是reg的操作还是会被优化。

通常这种写法一般用在对共享指针(指针被所有线程共享)的声明上,即这个指针变量有可能会被中断等函数修改。将其定义为volatile以后,编译器每次取指针变量的值的时候都会从内存中载入,这样即使这个变量已经被别的程序修改了当前函数用的时候也能得到修改后的值。不然会移植使用寄存器中保存的副本。

volatile uchar * volatile reg;

这样定义出来的指针就本身是个volatile的变量,又指向了volatile的数据内容。 


常见面试题

1、一个参数既可以是const还可以是volatile吗?

可以的,比如外部端口的值(大致可以理解为输入端口的高低电平),它的变化无需程序内的任何赋值语句就有可能改变的,这种变量就可以用volatile来修饰,complier不会优化掉它。const修饰的变量在程序里面是不能改变的,但是可以被程序外的东西修改,就象上面说的外部端口的值,如果仅仅使用const,有可能complier会优化掉这些变量,加上volatile就万无一失了。详见一个参数既可以是const还可以是volatile吗?解释为什么_朱智文的博客-CSDN博客

2、一个指针可以是volatile 吗?

可以的,当一个中服务子程序修改一个指向主函数的指针时。(也就是上面第三节中第二条代码框)。

  • 18
    点赞
  • 119
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
volatile关键字用于声明一个变量,确保多个线程对该变量的访问是可见的和原子的。当一个变量被声明为volatile后,Java内存模型会确保所有使用该变量的线程能看到相同的、一致的值。使用volatile关键字声明变量时,编译器对访问该变量的代码不再进行优化,这样可以提供对变量的稳定访问。未使用volatile修饰的变量,编译器可能会优化读取和存储,可能会使用寄存器中的值,导致不一致的现象。而使用volatile关键字能够保证直接访问内存,避免不一致问题的发生。总结起来,volatile关键字用法是确保多线程环境下对共享变量的可见性和原子性,避免出现线程间的数据不一致问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [java volatile关键字使用方法及注意事项](https://download.csdn.net/download/weixin_38697940/12775945)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [volatile 关键字的使用](https://blog.csdn.net/u011116085/article/details/129090784)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Kashine

你的鼓励将是我创作的动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值