OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 4444|回复: 8

大神们写程序的时候都用volatile这个关键字吗?

[复制链接]

68

主题

165

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
388
金钱
388
注册时间
2017-3-2
在线时间
156 小时
发表于 2018-12-2 17:44:18 | 显示全部楼层 |阅读模式
1金钱
大家在写程序时,尤其有系统的,比如uCOS等等,那种在多个任务都有使用的全局变量会加上 volatile 这个关键字吗?还有在中断中用的全局变量。
还有,如果全局变量使用的时候都用的取地址的方式比如 &var等等,还需要加volatile这个吗?我看好多程序都没加这个,尤其是uCOS源码里都没看见用这个关键字,不知道什么时候用。

最佳答案

查看完整内容[请看2#楼]

感觉你还没有理解为什么要有volatile这个关键词修饰符。简单举个不准确的例子吧:假设你做某个运算,有个变量A,程序为了加快运行速度,不仅仅将A放到内存中去,而是将A放到内存中去之后再在cache中放一份,运算的时候,读取写入A就直接操作cache中的就好了,这样cache比内存速度快,运算速度就能加快。而有的时候A不只是会被CPU改写,比如某些外设也会改写A,但是外设一般是无法改写cache的,只能改写内存中的A,因为cache是CPU的 ...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

33

主题

984

帖子

1

精华

论坛元老

Rank: 8Rank: 8

积分
8024
金钱
8024
注册时间
2014-8-13
在线时间
1595 小时
发表于 2018-12-2 17:44:19 | 显示全部楼层
感觉你还没有理解为什么要有volatile这个关键词修饰符。简单举个不准确的例子吧:假设你做某个运算,有个变量A,程序为了加快运行速度,不仅仅将A放到内存中去,而是将A放到内存中去之后再在cache中放一份,运算的时候,读取写入A就直接操作cache中的就好了,这样cache比内存速度快,运算速度就能加快。而有的时候A不只是会被CPU改写,比如某些外设也会改写A,但是外设一般是无法改写cache的,只能改写内存中的A,因为cache是CPU的一部分。这样,外设改写了A,可是CPU用来运算的A却没被改写,这样就会出现差错。甚至有时候A本身就是一个外设地址,就更加没办法映射到cache中去了。为了避免某些不能被映射的A映射到cache中去,就要用volatile修饰,这样,CPU每次读写A都去找内存中的地址,不去操作cache中的映像。如果你比较了解MCU的寄存器和外设配置,那么就很容易掌握什么时候该用volatile了。
回复

使用道具 举报

33

主题

984

帖子

1

精华

论坛元老

Rank: 8Rank: 8

积分
8024
金钱
8024
注册时间
2014-8-13
在线时间
1595 小时
发表于 2018-12-3 00:10:36 | 显示全部楼层
当然,也要看编译优化等级的,像是不优化时,没有任何变量被映射到cache中去,是否添加volatile就无所谓了,但是优化级别高了,为了保证程序不出错,就非常倚重volatile了。
回复

使用道具 举报

3

主题

238

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1366
金钱
1366
注册时间
2016-5-26
在线时间
1698 小时
发表于 2018-12-3 08:20:30 来自手机 | 显示全部楼层
mack13013 发表于 2018-12-2 17:44
感觉你还没有理解为什么要有volatile这个关键词修饰符。简单举个不准确的例子吧:假设你做某个运算,有个变 ...

c语言引入volatile关键字的年代应该比通用cpu引入cache要早的多……你是不是套用了其他语言的volatile……
mcu上c语言的volatile更多的是编译层面的事情吧,不标volatile的变量有可能被编译器直接搞没了,连cache操作都不会有的。
回复

使用道具 举报

6

主题

38

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
419
金钱
419
注册时间
2018-2-22
在线时间
81 小时
发表于 2018-12-3 09:04:50 | 显示全部楼层
K.O.Carnivist 发表于 2018-12-3 08:20
c语言引入volatile关键字的年代应该比通用cpu引入cache要早的多……你是不是套用了其他语言的volatile… ...

volatile简单来说就是让CPU字节从该变量的内存地址读取该变量值,而不是从CPU寄存器。
不加volatile的变量,某些情况下编译器会从CPU 寄存器读取该变量。
比如,
int n = 10, m = 0;

n = xxx操作;
。。。省略x行,这x行里都没对n在操作

m = n; 这里n的值一般是从CPU寄存器读取的,如果在那x行过程中,中断或者DMA对n值修改了就会导致m的值不是实际n的值



回复

使用道具 举报

6

主题

38

帖子

1

精华

中级会员

Rank: 3Rank: 3

积分
419
金钱
419
注册时间
2018-2-22
在线时间
81 小时
发表于 2018-12-3 09:10:16 | 显示全部楼层
mack13013 发表于 2018-12-2 17:44
感觉你还没有理解为什么要有volatile这个关键词修饰符。简单举个不准确的例子吧:假设你做某个运算,有个变 ...

volatile和cache没啥关系,简单来说volatile是CPU寄存器级别,cache是内存级别,volatile改变不来cache一致性问题,而且不是所有的CPU都带cache
回复

使用道具 举报

33

主题

984

帖子

1

精华

论坛元老

Rank: 8Rank: 8

积分
8024
金钱
8024
注册时间
2014-8-13
在线时间
1595 小时
发表于 2018-12-3 15:28:37 | 显示全部楼层
K.O.Carnivist 发表于 2018-12-3 08:20
c语言引入volatile关键字的年代应该比通用cpu引入cache要早的多……你是不是套用了其他语言的volatile… ...

嗯,我只是“简单举个不准确的例子吧:”,拿volatile在X86上某一个作用来说事的,不算完全介绍volatile作用。

实际上有没有cache,用寄存器还是用cache跟编译器优化等级、策略等有很多关系,我没办法非常“严密”、“完全"的给大家介绍volatile,所以只能拿“简单举个不准确的例子吧:”给大家打个比方。  大家不要在意这些细节,我并不是专业的。
回复

使用道具 举报

0

主题

62

帖子

0

精华

初级会员

Rank: 2

积分
111
金钱
111
注册时间
2018-11-29
在线时间
9 小时
发表于 2018-12-4 11:01:40 | 显示全部楼层
:@
回复

使用道具 举报

68

主题

165

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
388
金钱
388
注册时间
2017-3-2
在线时间
156 小时
 楼主| 发表于 2018-12-4 18:50:23 | 显示全部楼层
其实我就想知道为什么 uCOSIII里面没有用到这个关键字,系统源码里有很多全局变量,都没用这个关键字。不知道为什么,明显都是要跨线程使用的变量。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-9 02:10

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表