智能车制作

 找回密码
 注册

扫一扫,访问微社区

查看: 4227|回复: 12
打印 上一主题 下一主题

[分享] 【转载】 嵌入式C语言中const与volatile的用法详解

  [复制链接]

27

主题

615

帖子

0

精华

常驻嘉宾

Rank: 8Rank: 8

积分
3822

优秀会员奖章活跃会员奖章在线王奖章

威望
1583
贡献
697
兑换币
394
注册时间
2011-11-25
在线时间
771 小时
跳转到指定楼层
1#
发表于 2012-11-15 13:45:05 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
本帖最后由 jiyiboloann 于 2012-11-15 13:47 编辑

原文:
const最主要的特点就是只读,有常量、常量指针,如果不是特别小心的分析C语言语句的书写格式,再加上指针的使用,就特别容易弄错。
volatile关键字是一个类型修饰符,用它声明的类型变量表示可以被某些编译器未知的因素更改,比如:操作系统、硬件或者其它线程等。遇到这个关键字声明的变量,编译器对访问该变量的代码就不再进行优化,从而可以提供对特殊地址的稳定访问;如果不使用valatile,则编译器将对所声明的语句进行优化。之所以优化是因为访问寄存器要比访问内存单元快得多。但是优化之后容易出现问题,例如现在要直接对内存地址单元的内容修改,如果继续使用未经过valatile声明的变量,则读到的值有可能是寄存器中未经过修改的值,但本意是要读发生变化后的数值,所以会出现意想不到的错误。而经valatile声明的变量,每次访问该变量时都会从内存单元中重新读取。
const经常用于声明不希望被其它程序修改的常量;volatile经常用于声明因意外而可能发生改变的变量。

    下面具体分析两个变量的用法:

1、const
关键字const有什么含意?

我只要一听到被面试者说:“const意味着常数”,我就知道我正在和一个业余者打交道。其实只要能说出const意味着“只读”就可以了。尽管这个答案不是完全的答案,但我接受它作为一个正确的答案。如果应试者能正确回答这个问题,我将问他一个附加的问题:

下面的声明都是什么意思?
const int a;

int const a;

const int *a;

int * const a;

int const * a const;
前两个的作用是一样,a是一个常整型数;第三个意味着a是一个指向常整型数的指针(也就是,整型数是不可修改的,但指针可以);第四个意思a是一个指向整型数的常指针(也就是说,指针指向的整型数是可以修改的,但指针是不可修改的);最后一个意味着a是一个指向常整型数的常指针(也就是说,指针指向的整型数是不可修改的,同时指针也是不可修改的)。如果应试者能正确回答这些问题,那么他就给我留下了一个好印象。

2、volatile

关键字volatile有什么含意?并给出三个不同的例子。

一个定义为volatile的变量是说该变量可能会被意想不到地改变,这样,编译器就不会去假设该变量的值了。精确地说,优化器在用到该变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在寄存器里的备份。下面是volatile变量的几个例子:

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

(2) 一个中断服务子程序中会访问到的非自动变量

(2) 多线程应用中被几个任务共享的变量

回答不出这个问题的人是不会被雇佣的。我认为这是区分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;

}
补充:什么是自动变量
       自动变量就是指在函数内部定义使用的变量。它只允许在定义它的函数内部使用它。在函数外的其他任何地方都不能使用该变量。自动变量是局部变量,即它的区域性是在定义它的函数内部有效。当然这说明自动变量也没有链接性,因为它也不允许其它的文件访问它。由于自动变量在定义它的函数的外面的任何地方都是不可见的,所以允许我们在这个函数外的其它地方或者是其它的函数内部定义同名的变量,它们之间不会发生冲突的。因为它们都有自己的区域性,而其没有链接性(即:不允许其它的文件访问它)。

评分

1

查看全部评分

53

主题

2469

帖子

0

精华

跨届大侠

狙击手

Rank: 10Rank: 10Rank: 10

积分
11290

优秀会员奖章活跃会员奖章论坛元老奖章在线王奖章论坛骨干奖章资源大师奖章

威望
4701
贡献
3671
兑换币
2368
注册时间
2012-3-12
在线时间
1459 小时
2#
发表于 2012-11-15 13:48:17 | 只看该作者
:):):hug:
回复 支持 反对

使用道具 举报

17

主题

671

帖子

0

精华

跨届大侠

Rank: 10Rank: 10Rank: 10

积分
6187
QQ
威望
3024
贡献
1755
兑换币
1315
注册时间
2012-2-4
在线时间
704 小时
3#
发表于 2012-11-15 16:11:41 | 只看该作者
学习,这个还真不知道!
回复 支持 反对

使用道具 举报

27

主题

615

帖子

0

精华

常驻嘉宾

Rank: 8Rank: 8

积分
3822

优秀会员奖章活跃会员奖章在线王奖章

威望
1583
贡献
697
兑换币
394
注册时间
2011-11-25
在线时间
771 小时
4#
 楼主| 发表于 2012-11-15 16:43:21 | 只看该作者
青龙00 发表于 2012-11-15 16:11
学习,这个还真不知道!


回复 支持 反对

使用道具 举报

4

主题

397

帖子

0

精华

常驻嘉宾

Rank: 8Rank: 8

积分
3933
威望
1759
贡献
1014
兑换币
898
注册时间
2012-8-27
在线时间
580 小时
5#
发表于 2012-12-26 19:21:28 | 只看该作者
mark
回复 支持 反对

使用道具 举报

18

主题

690

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2173
威望
972
贡献
655
兑换币
673
注册时间
2012-11-20
在线时间
273 小时
毕业学校
安徽新华学院
6#
发表于 2012-12-27 00:06:32 | 只看该作者
学习了
回复 支持 反对

使用道具 举报

16

主题

1229

帖子

0

精华

跨届大侠

Rank: 10Rank: 10Rank: 10

积分
7678

优秀会员奖章论坛元老奖章活跃会员奖章在线王奖章

QQ
威望
3484
贡献
2020
兑换币
1673
注册时间
2011-9-8
在线时间
1087 小时
7#
发表于 2012-12-27 08:11:01 | 只看该作者
学习了
回复 支持 反对

使用道具 举报

1

主题

746

帖子

0

精华

常驻嘉宾

Rank: 8Rank: 8

积分
3319
QQ
威望
1414
贡献
501
兑换币
277
注册时间
2012-9-25
在线时间
702 小时
8#
发表于 2012-12-27 08:29:35 | 只看该作者
学习了,顶
回复 支持 反对

使用道具 举报

84

主题

707

帖子

2

精华

跨届大侠

Rank: 10Rank: 10Rank: 10

积分
7106

优秀会员奖章活跃会员奖章资源大师奖章论坛元老奖章论坛骨干奖章

威望
1736
贡献
4796
兑换币
434
注册时间
2012-2-15
在线时间
287 小时
毕业学校
西南交通大学
9#
发表于 2012-12-27 08:31:42 | 只看该作者
学习了,顶
回复 支持 反对

使用道具 举报

2

主题

13

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
威望
297
贡献
119
兑换币
102
注册时间
2012-8-2
在线时间
38 小时
毕业学校
合肥工业大学
10#
发表于 2012-12-27 09:40:23 | 只看该作者
谢谢楼主!!!顶呀!!!!!!
回复 支持 反对

使用道具 举报

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

本版积分规则

关于我们|联系我们|小黑屋|智能车制作 ( 黑ICP备2022002344号

GMT+8, 2024-6-26 23:57 , Processed in 0.086440 second(s), 34 queries , Gzip On.

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

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