volatile关键字

本文详细介绍了C语言中的volatile关键字,解释了其用途、原理,并通过实例展示了如何使用volatile关键字防止编译器优化,确保对特殊内存地址的稳定访问。文章还探讨了volatile关键字在中断服务程序、多任务环境及硬件寄存器等场景的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

volatile是C语言中的一个修饰符关键字。一个定义为volatile的变量是说这变量可能会被意想不到地改变,这样, 编译器就不会去假设这个变量的值了。当读取一个变量时,为提高存取速度,编译器优化时有时会先把变量读取到一个寄存器中; 以后,再取变量值时,就直接从寄存器中取值。 精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值, 而不是使用保存在寄存器里的备份或者被优化。 volatile应该解释为“直接存取原始内存地址”比较合适。 在嵌入式嵌入式开发中同硬件、中断、RTOS等打交道时都需要使用volatile变量。

volatile关键字是一种类型修饰符,用它声明的类型变量表示可以被某些编译器未知的因素更改,比如:操作系统、硬件或者其它线程等。遇到这个关键字声明的变量,编译器对访问该变量的代码就不再进行优化,从而可以提供对特殊地址的稳定访问。一句话:volatile可以防止编译器的某些优化。 
比如下面的代码:
#include 
int main(void)
{
    int x = 100;
    int a = x;
    printf("x=%d",a);
    //汇编语句改变内存中x的值,但编译器并不知道
    __asm
    {
        mov dword ptr[ebp-4],10h
    }
    int b = x;
    printf("x=%d",b);
    return 0;
}
上面代码,在debug版本(无优化)输出为:
x=100
x=16
在release版本(有优化)输出为:
x=100
x=100
这表明release模式下,编译器对代码进行了优化,一次读取了x的值,第二次没有更新x的取值,造成了错误的输出。假如对x使用了volatile关键字,那么输出结果又如何呢? 
int main(void)
{
    volatile int x = 100;
    int a = x;
    printf("x=%d",a);
    //汇编语句改变内存中x的值,但编译器并不知道
    __asm
    {
        mov dword ptr[ebp-4],10h
    }
    int b = x;
    printf("x=%d",b);

    return 0;
}
结果在debug和release版本下,都输出为:
x=100
x=16
这说明volatile让编译器“直接存取了x原始内存地址”。
另外一个通过volatile防止被编译器优化的例子:
for(int i=0; i<100000; i++)
; 面对这样的空循环,编译器肯定要把它优化掉,根本就不执行。但如果加上volatile:
for(volatile int i=0; i<100000; i++)
; 那么,编译器就不会对它进行优化,循环将会执行了。

一般说来,volatile用在如下的几个地方:
1、中断服务程序中修改的供其它程序检测的变量需要加volatile;
2、多任务环境下各任务间共享的标志应该加volatile;
3、存储器映射的硬件寄存器通常也要加volatile说明,因为每次对它的读写都可能有不同意义;

面试例题:
1). 一个参数既可以是const还可以是volatile吗?解释为什么。
可以。一个例子是只读的状态寄存器。它是volatile因为它可能被意想不到地改变。它是const因为程序不应该试图去修改它。
2). 一个指针可以是volatile 吗?解释为什么。
可以。一个例子是当一个中断服务子程序修改一个指向一个buffer的指针时。
3). 下面的函数被用来计算某个整数的平方,它能实现预期设计目标吗?如果不能,试回答存在什么问题
int square(volatile int *ptr)
{
    return ((*ptr) * (*ptr));
}
这段代码的目的是用来返指针*ptr指向值的平方,但是,由于*ptr指向一个volatile型参数,编译器将产生类似下面的代码:
int square(volatile int* &ptr)//这里参数应该申明为引用,不然函数体里只会使用副本,外部没法更改
{
    int a,b;
    a = *ptr;
    b = *ptr;
    return a*b;
}

由于*ptr的值可能在两次取值语句之间发生改变,因此a和b可能是不同的。结果,这段代码可能返回的不是你所期望的平方值!正确的代码如下: 
int square(volatile int*ptr)
{
    int a;
    a = *ptr;
    return a*a;
}
基于python实现的粒子群的VRP(车辆配送路径规划)问题建模求解+源码+项目文档+算法解析,适合毕业设计、课程设计、项目开发。项目源码已经过严格测试,可以放心参考并在此基础上延申使用,详情见md文档 算法设计的关键在于如何向表现较好的个体学习,标准粒子群算法引入惯性因子w、自我认知因子c1、社会认知因子c2分别作为自身、当代最优解和历史最优解的权重,指导粒子速度和位置的更新,这在求解函数极值问题时比较容易实现,而在VRP问题上,速度位置的更新则难以直接采用加权的方式进行,一个常见的方法是采用基于遗传算法交叉算子的混合型粒子群算法进行求解,这里采用顺序交叉算子,对惯性因子w、自我认知因子c1、社会认知因子c2则以w/(w+c1+c2),c1/(w+c1+c2),c2/(w+c1+c2)的概率接受粒子本身、当前最优解、全局最优解交叉的父代之一(即按概率选择其中一个作为父代,不加权)。 算法设计的关键在于如何向表现较好的个体学习,标准粒子群算法引入惯性因子w、自我认知因子c1、社会认知因子c2分别作为自身、当代最优解和历史最优解的权重,指导粒子速度和位置的更新,这在求解函数极值问题时比较容易实现,而在VRP问题上,速度位置的更新则难以直接采用加权的方式进行,一个常见的方法是采用基于遗传算法交叉算子的混合型粒子群算法进行求解,这里采用顺序交叉算子,对惯性因子w、自我认知因子c1、社会认知因子c2则以w/(w+c1+c2),c1/(w+c1+c2),c2/(w+c1+c2)的概率接受粒子本身、当前最优解、全局最优解交叉的父代之一(即按概率选择其中一个作为父代,不加权)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值