首页
新闻
论坛
群组
Blog
文档
下载
读书
Tag
网摘
搜索
.NET
Java
游戏
视频
人才
外包
培训
数据库
书店
程序员
欢迎您:
游客
| 退出
| 登录
注册
帮助
我的帖子
我参与的帖子
我的空间
我的网摘
CSDN
CSDN社区
C/C++
C语言
将帖子提前
放进我的网摘
推荐给好友
我要提问
帖子加分
生成帖子
置顶
推荐(加精)
取消推荐(加精)
锁定帖子
移动帖子
取消引用
结帖去...
管理菜单
页面风格切换
标准风格
老版本论坛
C语言中volatile是什么意思啊
[已结帖,结帖人:DuBin11]
加为好友
发送私信
在线聊天
DuBin11
该用户很懒,没有设置昵称
等级:
可用分等级:
乞丐
总技术分:
0
总技术分排名:
327011
结帖率:
48.31%
发表于:
2008-11-06 13:54:37
楼主
看书上写的有些模糊,哪位帮忙说明下
问题点数:
20
回复次数:
31
显示所有回复
显示星级回复
显示楼主回复
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
redleaves
程序员(http://hi.baidu.com/scu319hy)
等级:
可用分等级:
小地主
总技术分:
7520
总技术分排名:
2778
发表于:
2008-11-06 13:55:58
1
楼 得分:
0
这是C++的关键字,是容易改变的意思.
就是告诉编译器,这个对象可能在一些情况被其它的代码改变.在优化时要注意.不要做缓存等...
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
lbh2001
C Fans
等级:
可用分等级:
富农
总技术分:
7007
总技术分排名:
6635
发表于:
2008-11-06 13:56:14
2
楼 得分:
0
C语言关键字volatile(易变的)表示不经过赋值,其值也可能被改变
(例如,表示时钟的变量、表示通行端口的变量等)
会阻止编译器优化
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
xhs_lh04
无头像
等级:
可用分等级:
富农
总技术分:
7475
总技术分排名:
3009
发表于:
2008-11-06 13:57:16
3
楼 得分:
0
优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
longbao
等级:
可用分等级:
长工
总技术分:
27
总技术分排名:
247291
发表于:
2008-11-06 13:57:37
4
楼 得分:
20
volatile总是与优化有关,编译器有一种技术叫做数据流分析,分析程序中的变量在哪里赋值、在哪里使用、在哪里失效,分析结果可以用于常量合并,常量传播等优化,进一步可以死代码消除。但有时这些优化不是程序所需要的,这时可以用volatile关键字禁止做这些优化,volatile的字面含义是易变的,它有下面的作用:
1 不会在两个操作之间把volatile变量缓存在寄存器中。在多任务、中断、甚至setjmp环境下,变量可能被其他的程序改变,编译器 自己无法知道,volatile就是告诉编译器这种情况。
2 不做常量合并、常量传播等优化,所以像下面的代码:
volatile int i = 1;
if (i > 0) ...
if的条件不会当作无条件真。
3 对volatile变量的读写不会被优化掉。如果你对一个变量赋值但后面没用到,编译器常常可以省略那个赋值操作,然而对Memory Mapped IO的处理是不能这样优化的。
有人说volatile可以保证对内存操作的原子性,这种说法不大准确,其一,x86需要LOCK前缀才能在SMP下保证原子性,其二,RISC根本不能对内存直接运算,要保证原子性得用别的方法,如atomic_inc。
对于jiffies,它已经声明为volatile变量,我认为直接用jiffies++就可以了,没必要用那种复杂的形式,因为那样也不能保证原子性。
你可能不知道在Pentium及后续CPU中,下面两组指令
inc jiffies
;;
mov jiffies, %eax
inc %eax
mov %eax, jiffies
作用相同,但一条指令反而不如三条指令快。
------------------------------------------------------------------------------------------------------------
一般这个修饰符用来告知编译器,被修饰的变量是个“易变的”变量(volatile的本意是“易变的”),防止编译器进行优化。将变量加上volatile修饰,则编译器保证对此变量的读写操作都不会被优化。
1、中断服务程序中修改的供其它程序检测的变量需要加volatile;
2、多任务环境下各任务间共享的标志应该加volatile;
3、存储器映射的硬件寄存器通常也要加volatile说明,因为每次对它的读写都可能由不同意义
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
bxhzct
bxhzct
等级:
可用分等级:
中农
总技术分:
318
总技术分排名:
52293
发表于:
2008-11-06 16:26:31
5
楼 得分:
0
防止进入寄存器中优化,4楼说的 很详细,轻参考!!
这个变量在嵌入式编程中用的很多
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
cyj626
人在江湖,江湖却没有我的传说-_-!
等级:
可用分等级:
中农
总技术分:
584
总技术分排名:
31464
发表于:
2008-11-06 16:27:56
6
楼 得分:
0
引用 2 楼 lbh2001 的回复:
C语言关键字volatile(易变的)表示不经过赋值,其值也可能被改变
(例如,表示时钟的变量、表示通行端口的变量等)
会阻止编译器优化
up
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
jackzhhuang
专业:C/C++,SDK,MFC,masm,dll,com
等级:
可用分等级:
富农
总技术分:
1510
总技术分排名:
16104
发表于:
2008-11-06 18:46:00
7
楼 得分:
0
表示这个值随时可能改变,这样多线程的时候,线程每次访问这个值的时候都会检查一次,不再做优化。
我记得windows多线程编程一书中有例子,一时记不起来了。
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
czbever
骑着蜗牛看世界
等级:
可用分等级:
中农
总技术分:
264
总技术分排名:
59364
发表于:
2008-11-06 18:51:12
8
楼 得分:
0
2008-10-27 11:09volatile关键字是一种类型修饰符,用它声明的类型变量表示可以被某些编译器未知的因素更改,比如:操作系统、硬件或者其它线程等。遇到这个关键字声明的变量,编译器对访问该变量的代码就不再进行优化,从而可以提供对特殊地址的稳定访问。
使用该关键字的例子如下:
int volatile nVint;
当要求使用volatile 声明的变量的值的时候,系统总是重新从它所在的内存读取数据,即使它前面的指令刚刚从该处读取过数据。而且读取的数据立刻被保存。
例如:
volatile int i=10;
int a = i;
...
//其他代码,并未明确告诉编译器,对i进行过操作
int b = i;
volatile 指出 i是随时可能发生变化的,每次使用它的时候必须从i的地址中读取,因而编译器生成的汇编代码会重新从i的地址读取数据放在b中。而优化做法是,由于编译器发现两次从i读数据的代码之间的代码没有对i进行过操作,它会自动把上次读的数据放在b中。而不是重新从i里面读。这样以来,如果i是一个寄存器变量或者表示一个端口数据就容易出错,所以说volatile可以保证对特殊地址的稳定访问。
####################典型问题######################
关键字volatile有什么含意?并给出三个不同的例子。
一个定义为volatile的变量是说这变量可能会被意想不到地改变,这样,编译器就不会去假设这个变量的值了。精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。下面是volatile变量的几个例子:
1). 并行设备的硬件寄存器(如:状态寄存器)
2). 一个中断服务子程序中会访问到的非自动变量(Non-automatic variables)
3). 多线程应用中被几个任务共享的变量
回答不出这个问题的人是不会被雇佣的。我认为这是区分C程序员和嵌入式系统程序员的最基本的问题。嵌入式系统程序员经常同硬件、中断、RTOS等等打交道,所用这些都要求volatile变量。不懂得volatile内容将会带来灾难。
假设被面试者正确地回答了这是问题(嗯,怀疑这否会是这样),我将稍微深究一下,看一下这家伙是不是直正懂得volatile完全的重要性。
1). 一个参数既可以是const还可以是volatile吗?解释为什么。
2). 一个指针可以是volatile 吗?解释为什么。
3). 下面的函数有什么错误:
int square(volatile int *ptr)
{
return *ptr * *ptr;
}
下面是答案:
1). 是的。一个例子是只读的状态寄存器。它是volatile因为它可能被意想不到地改变。它是const因为程序不应该试图去修改它。
2). 是的。尽管这并不很常见。一个例子是当一个中服务子程序修该一个指向一个buffer的指针时。
3). 这段代码的有个恶作剧。这段代码的目的是用来返指针*ptr指向值的平方,但是,由于*ptr指向一个volatile型参数,编译器将产生类似下面的代码:
int square(volatile int *ptr)
{
int a,b;
a = *ptr;
b = *ptr;
return a * b;
}
由于*ptr的值可能被意想不到地该变,因此a和b可能是不同的。结果,这段代码可能返不是你所期望的平方值!正确的代码如下:
long square(volatile int *ptr)
{
int a;
a = *ptr;
return a * a;
}
************************补充*******************************
volatile的本意是“易变的” 由于访问寄存器的速度要快过RAM,所以编译器一般都会作减少存取外部RAM的优化。比如:
static int i=0; //应该为static volatile int i=0;
int main(void)
{
...
while (1)
{
if (i) dosomething();
}
}
/* Interrupt service routine. */
void ISR_2(void)
{
i=1;
}
程序的本意是希望ISR_2中断产生时,在main当中调用dosomething函数,但是,由于编译器判断在main函数里面没有修改过i,因此可能只执行一次对从i到某寄存器的读操作,然后每次if判断都只使用这个寄存器里面的“i副本”,导致dosomething永远也不会被调用。
如果将将变量加上volatile修饰,则编译器保证对此变量的读写操作都不会被优化(肯定执行)。此例中i也应该如此说明。
一般说来,volatile用在如下的几个地方:
1、中断服务程序中修改的供其它程序检测的变量需要加volatile;
2、多任务环境下各任务间共享的标志应该加volatile;
3、存储器映射的硬件寄存器通常也要加volatile说明,因为每次对它的读写都可能由不同意义;
volatile表示变量的内容可能在程序未知的情况下被改变
比如,它对应的内存地址的内容被中断函数,或者其他的进程所改变
这种类型的变量,程序执行的时候不会放到cache当中预取,而是每次用到的时候直接取得
比如,你在c中间写这样的程序
for (int i=0; i <100000;i++);
空循环,什么也不做
这个东西就会被优化调,如果在int前面加入这个标记则不会被优化的,因为i每次的变化不一定++也虚在循环中间被别的程序所改变
在linux的source code(linux/mm/memory.c)中有这样两句:
volatile void do_exit(long code);
static inline volatile void oom(void)
{
printk("out of memory\n\r");
do_exit(SIGSEGV);
}
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
orler
该用户很懒,没有设置昵称
等级:
可用分等级:
长工
总技术分:
10
总技术分排名:
220639
发表于:
2008-11-06 20:34:06
9
楼 得分:
0
表示该值可能在编译器未监视到的情况下改变,告诉编译器不要武断的应用该变量的代码优化。
有三种情况:
1,并行设备的硬件寄存器。如状态寄存器
2,中断服务子程序访问到的非自动变量
3,多线程应用中被几个变量共享的变量
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
lann64
昆仑大鹏@迦楼罗
等级:
可用分等级:
富农
总技术分:
14626
总技术分排名:
1583
发表于:
2008-11-06 20:45:13
10
楼 得分:
0
能关注到这个,说明功力已经晋级了,呵呵
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
insulted
蓝田沧海
等级:
可用分等级:
中农
总技术分:
125
总技术分排名:
80731
发表于:
2008-11-06 20:57:31
11
楼 得分:
0
路过,学习!
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
redleaves
程序员(http://hi.baidu.com/scu319hy)
等级:
可用分等级:
小地主
总技术分:
7520
总技术分排名:
2778
发表于:
2008-11-06 22:08:38
12
楼 得分:
0
引用 10 楼 lann64 的回复:
能关注到这个,说明功力已经晋级了,呵呵
等发现原来这个也不过是个摆设的时候...
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
insulted
蓝田沧海
等级:
可用分等级:
中农
总技术分:
125
总技术分排名:
80731
发表于:
2008-11-06 22:11:55
13
楼 得分:
0
8楼的解释很好很强大!
最后的一部分:
在linux的source code(linux/mm/memory.c)中有这样两句:
volatile void do_exit(long code);
static inline volatile void oom(void)
{
printk("out of memory\n\r");
do_exit(SIGSEGV);
}
在下愚钝,没有看明白,还望路过的高人指点迷津!
先有礼了!
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
leank
该用户很懒,没有设置昵称
等级:
可用分等级:
中农
总技术分:
208
总技术分排名:
66092
发表于:
2008-11-06 22:39:24
14
楼 得分:
0
恶补
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
hairetz
每天保持一颗轻松幽默的心.
等级:
可用分等级:
长工
总技术分:
3583
总技术分排名:
15042
发表于:
2008-11-07 10:20:09
15
楼 得分:
0
貌似多线程的时候有用到,很多信号量就是volatile。
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
snailman
一辈子
等级:
可用分等级:
富农
总技术分:
247
总技术分排名:
53856
发表于:
2008-11-07 11:35:10
16
楼 得分:
0
[color=#FF0000]volatile 修饰的变量 不会呗放放到寄存器里,需要的时候去读内存里的值,如果不加这个关键字,那么这个变量就可能呗放到寄存器里加快访问速度。一般在多线程环境下使用,因为有些变量需要时候改变。[/color]
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
jackzhizhi
该用户很懒,没有设置昵称
等级:
可用分等级:
乞丐
总技术分:
0
总技术分排名:
327011
发表于:
2008-11-07 23:32:26
17
楼 得分:
0
引用 2 楼 lbh2001 的回复:
C语言关键字volatile(易变的)表示不经过赋值,其值也可能被改变
(例如,表示时钟的变量、表示通行端口的变量等)
会阻止编译器优化
Right
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
jackzhizhi
该用户很懒,没有设置昵称
等级:
可用分等级:
乞丐
总技术分:
0
总技术分排名:
327011
发表于:
2008-11-07 23:34:43
18
楼 得分:
0
引用 9 楼 orler 的回复:
表示该值可能在编译器未监视到的情况下改变,告诉编译器不要武断的应用该变量的代码优化。
有三种情况:
1,并行设备的硬件寄存器。如状态寄存器
2,中断服务子程序访问到的非自动变量
3,多线程应用中被几个变量共享的变量
被"被几个变量共享的变量"
应是"被几个任务共享的变量"
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
jillnicky
吉尔
等级:
可用分等级:
乞丐
总技术分:
6176
总技术分排名:
3656
发表于:
2008-11-07 23:40:15
19
楼 得分:
0
有答案了
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
traceless
老铅:一个纯粹的csdner
等级:
可用分等级:
贫农
总技术分:
50
总技术分排名:
197778
发表于:
2008-11-07 23:57:06
20
楼 得分:
0
引用 12 楼 redleaves 的回复:
引用 10 楼 lann64 的回复:
能关注到这个,说明功力已经晋级了,呵呵
等发现原来这个也不过是个摆设的时候...
哈哈 高手的对话,都很有理呀
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
hqin6
!独行
等级:
可用分等级:
富农
总技术分:
18762
总技术分排名:
817
发表于:
2008-11-08 00:30:36
21
楼 得分:
0
C/C++ code
就是说,这个值是不可靠的! 可靠的在原始位置,而不是寄存器!
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
zwjiong
等级:
可用分等级:
贫农
总技术分:
7
总技术分排名:
247291
发表于:
2008-11-08 09:35:33
22
楼 得分:
0
在做嵌入式系统的时候特别要关注这个~~~因为中断会随时改变变量的值,而如果只是读变量的缓存值就会出错
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
zhxin999
等级:
可用分等级:
贫农
总技术分:
57
总技术分排名:
123645
发表于:
2008-11-08 11:42:33
23
楼 得分:
0
取消编译器优化
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
cxxer
thinking
等级:
可用分等级:
掌柜
总技术分:
492
总技术分排名:
33771
发表于:
2008-11-08 13:50:10
24
楼 得分:
0
易失性变量.
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
suxpert
连长
等级:
可用分等级:
中农
总技术分:
374
总技术分排名:
41946
发表于:
2008-11-08 13:53:23
25
楼 得分:
0
ls相当详细,看来又晚了一步……
嵌入式,驱动等等中经常会用到。
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
godkiller007
上帝杀手007
等级:
可用分等级:
贫农
总技术分:
0
总技术分排名:
327011
发表于:
2008-11-09 00:41:20
26
楼 得分:
0
mark
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
rlj021
*****
等级:
可用分等级:
掌柜
总技术分:
457
总技术分排名:
38325
发表于:
2008-11-10 19:28:18
27
楼 得分:
0
顶……
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
xxyzily
该用户很懒,没有设置昵称
等级:
可用分等级:
短工
总技术分:
0
总技术分排名:
327011
发表于:
2008-11-20 16:54:59
28
楼 得分:
0
回帖好的帖子是一种美德~!!谢了,学习了
修改
删除
举报
引用
回复
加为好友
发送私信
在线聊天
whsfer
Alwin
等级:
可用分等级:
贫农
总技术分:
12
总技术分排名:
251750
发表于:
2008-12-08 11:33:09
29
楼 得分:
0
学习了
修改