首页 新闻 论坛 群组 Blog 文档 下载 读书 Tag 网摘 搜索 .NET Java 游戏 视频 人才 外包 培训 数据库 书店 程序员
中国软件网
欢迎您:游客 | 登录 注册 帮助
  • C语言中volatile是什么意思啊 [已结帖,结帖人:DuBin11]
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • DuBin11
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    • 结帖率:
    发表于:2008-11-06 13:54:37 楼主
    看书上写的有些模糊,哪位帮忙说明下
    20  修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • redleaves
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 13:55:581楼 得分:0
    这是C++的关键字,是容易改变的意思.
    就是告诉编译器,这个对象可能在一些情况被其它的代码改变.在优化时要注意.不要做缓存等...
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • lbh2001
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 13:56:142楼 得分:0
    C语言关键字volatile(易变的)表示不经过赋值,其值也可能被改变
    (例如,表示时钟的变量、表示通行端口的变量等)
    会阻止编译器优化
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • xhs_lh04
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 13:57:163楼 得分:0
    优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • longbao
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 13:57:374楼 得分: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
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 16:26:315楼 得分:0
    防止进入寄存器中优化,4楼说的 很详细,轻参考!!
    这个变量在嵌入式编程中用的很多
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • cyj626
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 16:27:566楼 得分:0
    引用 2 楼 lbh2001 的回复:
    C语言关键字volatile(易变的)表示不经过赋值,其值也可能被改变
    (例如,表示时钟的变量、表示通行端口的变量等)
    会阻止编译器优化

    up
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • jackzhhuang
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 18:46:007楼 得分:0
    表示这个值随时可能改变,这样多线程的时候,线程每次访问这个值的时候都会检查一次,不再做优化。

    我记得windows多线程编程一书中有例子,一时记不起来了。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • czbever
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 18:51:128楼 得分: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
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 20:34:069楼 得分:0
    表示该值可能在编译器未监视到的情况下改变,告诉编译器不要武断的应用该变量的代码优化。
    有三种情况:
    1,并行设备的硬件寄存器。如状态寄存器
    2,中断服务子程序访问到的非自动变量
    3,多线程应用中被几个变量共享的变量
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • lann64
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 20:45:1310楼 得分:0
    能关注到这个,说明功力已经晋级了,呵呵
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • insulted
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 20:57:3111楼 得分:0
    路过,学习!
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • redleaves
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 22:08:3812楼 得分:0
    引用 10 楼 lann64 的回复:
    能关注到这个,说明功力已经晋级了,呵呵

    等发现原来这个也不过是个摆设的时候...
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • insulted
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 22:11:5513楼 得分: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
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-06 22:39:2414楼 得分:0
    恶补
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • hairetz
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-07 10:20:0915楼 得分:0
    貌似多线程的时候有用到,很多信号量就是volatile。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • snailman
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-07 11:35:1016楼 得分:0
    [color=#FF0000]volatile 修饰的变量 不会呗放放到寄存器里,需要的时候去读内存里的值,如果不加这个关键字,那么这个变量就可能呗放到寄存器里加快访问速度。一般在多线程环境下使用,因为有些变量需要时候改变。[/color]
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • jackzhizhi
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-07 23:32:2617楼 得分:0
    引用 2 楼 lbh2001 的回复:
    C语言关键字volatile(易变的)表示不经过赋值,其值也可能被改变
    (例如,表示时钟的变量、表示通行端口的变量等)
    会阻止编译器优化
    Right
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • jackzhizhi
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-07 23:34:4318楼 得分:0
    引用 9 楼 orler 的回复:
    表示该值可能在编译器未监视到的情况下改变,告诉编译器不要武断的应用该变量的代码优化。
    有三种情况:
    1,并行设备的硬件寄存器。如状态寄存器
    2,中断服务子程序访问到的非自动变量
    3,多线程应用中被几个变量共享的变量

    被"被几个变量共享的变量"
    应是"被几个任务共享的变量"
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • jillnicky
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-07 23:40:1519楼 得分:0
    有答案了
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • traceless
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-07 23:57:0620楼 得分:0
    引用 12 楼 redleaves 的回复:
    引用 10 楼 lann64 的回复:
    能关注到这个,说明功力已经晋级了,呵呵

    等发现原来这个也不过是个摆设的时候...


    哈哈 高手的对话,都很有理呀
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • hqin6
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-08 00:30:3621楼 得分:0
    C/C++ code
    就是说,这个值是不可靠的! 可靠的在原始位置,而不是寄存器!
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • zwjiong
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-08 09:35:3322楼 得分:0
    在做嵌入式系统的时候特别要关注这个~~~因为中断会随时改变变量的值,而如果只是读变量的缓存值就会出错
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • zhxin999
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-08 11:42:3323楼 得分:0
    取消编译器优化
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • cxxer
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-08 13:50:1024楼 得分:0
    易失性变量.
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • suxpert
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-08 13:53:2325楼 得分:0
    ls相当详细,看来又晚了一步……

    嵌入式,驱动等等中经常会用到。
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • godkiller007
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-09 00:41:2026楼 得分:0
    mark
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • rlj021
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-10 19:28:1827楼 得分:0
    顶……
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • xxyzily
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-11-20 16:54:5928楼 得分:0
    回帖好的帖子是一种美德~!!谢了,学习了
    修改 删除 举报 引用 回复
    进入用户个人空间
    加为好友
    发送私信
    在线聊天
    • whsfer
    • 等级:
    • 可用分等级:
    • 总技术分:
    • 总技术分排名:
    发表于:2008-12-08 11:33:0929楼 得分:0
    学习了
    修改