【转载】变量的存储类别
转载自:http://edu.cnzz.cn/852996fd.html
一、动态存储方式与静态存储方式
上一节已介绍了变量的一种属性——作用域,作用域是从空间的角度来分析的,分为全局变量和局部变量。
变 量还有另一种属性——存储期(storage duration,也称生命期)。存储期是指变量在内存中的存在期间。这是从变量值存在的时间角度来分析的。存储期可以分为静态存储期(static storage duration)和动态存储期(dynamic storage duration)。这是由变量的静态存储方式和动态存储方式决定的。
所谓静态存储方式是指在程序运行期间,系统对变量分配固定的存储空间。而动态存储方式则是在程序运行期间,系统对变量动态地分配存储空间。
先看一下内存中的供用户使用的存储空间的情况。这个存储空间可以分为三部分,即:
(1) 程序区
(2) 静态存储区
(3) 动态存储区
数据分别存放在静态存储区和动态存储区中。全局变量全部存放在静态存储区中,在程序开始执行时给全局变量分配存储单元,程序执行完毕就释放这些空间。在程序执行过程中它们占据固定的存储单元,而不是动态地进行分配和释放。
在动态存储区中存放以下数据:
①函数形式参数。在调用函数时给形参分配存储空间。
②函数中的自动变量(未加static声明的局部变量,详见后面的介绍)。
③函数调用时的现场保护和返回地址等。
对以上这些数据,在函数调用开始时分配动态存储空间,函数结束时释放这些空间。在程序执行过程中,这种分配和释放是动态的,如果在一个程序中两次调用同一函数,则要进行两次分配和释放,而两次分配给此函数中局部变量的存储空间地址可能是不相同的。
如果在一个程序中包含若干个函数,每个函数中的局部变量的存储期并不等于整个程序的执行周期,它只是整个程序执行周期的一部分。根据函数调用的情况,系统对局部变量动态地分配和释放存储空间。
在 C++中变量除了有数据类型的属性之外,还有存储类别(storage class) 的属性。存储类别指的是数据在内存中存储的方法。存储方法分为静态存储和动态存储两大类。具体包含4种:自动的(auto)、静态的(static)、寄 存器的(register)和外部的(extern)。根据变量的存储类别,可以知道变量的作用域和存储期。
二、自动变量
函 数中的局部变量,如果不用关键字static加以声明,编译系统对它们是动态地分配存储空间的。函数的形参和在函数中定义的变量(包括在复合语句中定义的 变量)都属此类。在调用该函数时,系统给形参和函数中定义的变量分配存储空间,数据存储在动态存储区中。在函数调用结束时就自动释放这些空间。如果是在复 合语句中定义的变量,则在变量定义时分配存储空间,在复合语句结束时自动释放空间。因此这类局部变量称为自动变量(auto variable)。自动变量用关键字auto作存储类别的声明。例如:
int f(int a) //定义f函数,a为形参
{
auto int b,c=3; //定义b和c为整型的自动变量
┆
}
存 储类别auto和数据类型int的顺序任意。关键字auto可以省略,如果不写auto,则系统把它默认为自动存储类别,它属于动态存储方式。程序中大多 数变量属于自动变量。本书前面各章所介绍的例子中,在函数中定义的变量都没有声明为auto,其实都默认指定为自动变量。在函数体中以下两种写法作用相 同:
① auto int b,c=3;
② int b,c=3;
三、用static声明静态局部变量
有时希望函数中的局部变量的值在函数调用结束后不消失而保留原值,即其占用的存储单元不释放,在下一次该函数调用时,该变量保留上一次函数调用结束时的值。这时就应该指定该局部变量为静态局部变量(static local variable)。
例4.12 静态局部变量的值。
#include <iostream>
using namespace std;
int f(int a) //定义f函数,a为形参
{
auto int b=0; //定义b为自动变量
static int c=3;//定义c为静态局部变量
b=b+1;
c=c+1;
return a+b+c;
}
int main( )
{
int a=2,i;
for(i=0;i<3;i++)
cout<<f(a)<<″ ″;
cout<<endl;
return 0;
}
运行结果为
7 8 9
先后3次调用f函数时,b和c的值如书中表4.1所示。
对静态局部变量的说明:
1.静态局部变量在静态存储区内分配存储单元。在程序整个运行期间都不释放。而自动变量(即动态局部变量)属于动态存储类别,存储在动态存储区空间(而不是静态存储区空间),函数调用结束后即释放。
2. 为静态局部变量赋初值是在编译时进行值的,即只赋初值一次,在程序运行时它已有初值。以后每次调用函数时不再重新赋初值而只是保留上次函数调用结束时的 值。而为自动变量赋初值,不是在编译时进行的,而是在函数调用时进行,每调用一次函数重新给一次初值,相当于执行一次赋值语句。
3.如果在 定义局部变量时不赋初值的话,对静态局部变量来说,编译时自动赋初值0(对数值型变量)或空字符(对字符型变量)。而对自动变量来说,如果不赋初值,则它 的值是一个不确定的值。这是由于每次函数调用结束后存储单元已释放,下次调用时又重新另分配存储单元,而所分配的单元中的值是不确定的。
4.虽然静态局部变量在函数调用结束后仍然存在,但其他函数是不能引用它的,也就是说,在其他函数中它是“不可见”的。
在什么情况下需要用局部静态变量呢?
1.需要保留函数上一次调用结束时的值。例如可以用下例中的方法求n!。
例4.13 输出1~5的阶乘值(即1!,2!,3!,4!,5!)。
#include <iostream>
using namespace std;
int fac(int); //函数声明
int main( )
{
int i;
for(i=1;i<=5;i++)
cout<<i<<″!=″<<fac(i)<<endl;
return 0;
}
int fac(int n)
{
static int f=1; //f为静态局部变量,函数结束时f的值不释放
f=f*n; //在f原值基础上乘以n
return f;
}
运行结果为
1!=1
2!=2
3!=6
4!=24
5!=120
每次调用fac(i),就输出一个i,同时保留这个i!的值,以便下次再乘(i+1)。
2.如果初始化后,变量只被引用而不改变其值,则这时用静态局部变量比较方便,以免每次调用时重新赋值。 但是应该看到,用静态存储要多占内存,而且降低了程序的可读性,当调用次数多时往往弄不清静态局部变量的当前值是什么。因此,如不必要,不要多用静态局部变量。
四、用register声明寄存器变量
一般情况下,变量的值是存放在内存中的。当程序中用到哪一个变量的值时,由控制器发出指令将内存中该变量的值送到CPU中的运算器。经过运算器进行运算,如果需要存数,再从运算器将数据送到内存存放。如图4.15所示。
为提高执行效率,C++允许将局部变量的值放在CPU中的寄存器中,需要用时直接从寄存器取出参加运算,不必再到内存中去存取。这种变量叫做寄存器变量,用关键字register作声明。例如,可以将例4.14中的fac函数改写如下:
int fac(int n)
{
register int i,f=1; //定义i和f是寄存器变量
for(i=1;i<=n;i++) f=f*i;
return f;
}
定义f和i是存放在寄存器的局部变量,如果n的值大,则能节约许多执行时间。
在程序中定义寄存器变量对编译系统只是建议性(而不是强制性)的。当今的优化编译系统能够识别使用频繁的变量,自动地将这些变量放在寄存器中。
五、用extern声明外部变量
全局变量(外部变量)是在函数的外部定义的,它的作用域为从变量的定义处开始,到本程序文件的末尾。在此作用域内,全局变量可以为本文件中各个函数所引用。编译时将全局变量分配在静态存储区。
有时需要用extern来声明全局变量,以扩展全局变量的作用域。
1. 在一个文件内声明全局变量
如 果外部变量不在文件的开头定义,其有效的作用范围只限于定义处到文件终了。如果在定义点之前的函数想引用该全局变量,则应该在引用之前用关键字 extern对该变量作外部变量声明,表示该变量是一个将在下面定义的全局变量。有了此声明,就可以从声明处起,合法地引用该全局变量,这种声明称为提前 引用声明。
例4.14 用extern对外部变量作提前引用声明,以扩展程序文件中的作用域。
#include <iostream>
using namespace std;
int max(int,int); //函数声明
void main( )
{
extern int a,b;//对全局变量a,b作提前引用声明
cout<<max(a,b)<<endl;
}
int a=15,b=-7;//定义全局变量a,b
int max(int x,int y)
{
int z;
z=x>y?x:y;
return z;
}
运行结果如下:
15
在 main后面定义了全局变量a,b,但由于全局变量定义的位置在函数main之后,因此如果没有程序的第5行,在main函数中是不能引用全局变量a和b 的。现在我们在main函数第2行用extern对a和b作了提前引用声明,表示a和b是将在后面定义的变量。这样在main函数中就可以合法地使用全局 变量a和b了。如果不作extern声明,编译时会出错,系统认为a和b未经定义。一般都把全局变量的定义放在引用它的所有函数之前,这样可以避免在函数 中多加一个extern声明。
2. 在多文件的程序中声明外部变量
如果一个程序包含两个文件,在两个文件中都要用到同一个外部变量num,不能分别在两个文件中各自定义一个外部变量num。正确的做法是:在任一个文件中定义外部变量num,而在另一文件中用extern对num作外部变量声明。即
extern int num;
编 译系统由此知道num是一个已在别处定义的外部变量,它先在本文件中找有无外部变量num,如果有,则将其作用域扩展到本行开始(如上节所述),如果本文 件中无此外部变量,则在程序连接时从其他文件中找有无外部变量num,如果有,则把在另一文件中定义的外部变量num的作用域扩展到本文件,在本文件中可 以合法地引用该外部变量num。
分析下例:
file1.cppfile2.cpp
extern int a,b; int a=3,b=4;
int main( ) ┆
{
cout<<a<<″,″<<b<<endl;
return 0;
}
用extern扩展全局变量的作用域,虽然能为程序设计带来方便,但应十分慎重,因为在执行一个文件中的函数时,可能会改变了该全局变量的值,从而会影响到另一文件中的函数执行结果。
六、用static声明静态外部变量
有时在程序设计中希望某些外部变量只限于被本文件引用,而不能被其他文件引用。这时可以在定义外部变量时加一个static声明。例如:
file1.cppfile2.cpp
static int a=3;extern int a;
int main ( ) int fun (int n)
{{ ┆
┆a=a*n;
┆
}}
这种加上static声明、只能用于本文件的外部变量(全局变量)称为静态外部变量。这就为程序的模块化、通用性提供了方便。如果已知道其他文件不需要引用本文件的全局变量,可以对本文件中的全局变量都加上static,成为静态外部变量,以免被其他文件误用。
需要指出,不要误认为用static声明的外部变量才采用静态存储方式(存放在静态存储区中),而不加static的是动态存储(存放在动态存储区)。实际上,两种形式的外部变量都用静态存储方式,只是作用范围不同而已,都是在编译时分配内存的。
转载于:https://www.cnblogs.com/pengdonglin137/p/3239531.html
【转载】变量的存储类别相关推荐
- C语言中局部变量和全局变量 变量的存储类别
C语言中局部变量和全局变量 变量的存储类别(static,extern,auto,register) 局部变量和全局变量 在讨论函数的形参变量时曾经提到,形参变量只在被调用期间才分配内存单元,调用 ...
- c语言作用域有哪两种变量,2017年计算机二级C语言字考点归纳:变量的存储类别、作用域及生存期...
7.7 变量的存储类别.作用域及生存期 1.变量的存储类别 在C语言中,有两类存储类别:自动类别及静态类别. 有4个与两种存储类别有关的说明符:auto(自动).register(寄存器).stati ...
- 存储过程debug值not a variable_C语言变量的存储类别
在程序中经常会使用到变量,在C程序中可以选择变量的不同存储形式,其存储类别分为静态存储和动态存储.可以通过存储类别修饰符来告诉编译器要处理什么样的类型变量,具体主要有自动(auto).静态(stati ...
- C语言变量的存储类别和生存期
C语言变量的存储类别和生存期 我们知道,变量是有数据类型的,用来说明他占用多大的内存空间,可以进行什么操作. 除了数据类型,变量还有一个属性,称为"存储类别".存储类别就是变量在内 ...
- 变量的存储类别以及作用空间
从变量值存在的时间角度划分,可以把变量划为静态存储空间和动态存储空间. 所谓静态存储空间是指在程序运行期间分配固定的存储空间的方式.而动态存储空间是在程序运行期间根据需要动态分配存储空间. 内存中供用 ...
- c语言中变量有几种存储方式,C语言变量的存储类别有哪些详细资料介绍
描述 C语言中的变量根据其生存周期,可以分为静态存储方式和动态存储方式. 静态存储方式:是指在程序运行期间分配固定的存储空间的方式.静态存储区中存放了在整个程序执行过程中都存在的变量,如全局变量. 动 ...
- 动态内存分配及变量存储类别(第二部分)
5. C语言变量的存储类别和生存期 我们知道,变量是有数据类型的,用以说明它占用多大的内存空间,可以进行什么样的操作. 除了数据类型,变量还有一个属性,称为"存储类别".存储类别就 ...
- C++变量作用域、生存期、存储类别
写C.C++代码的小伙伴一定在头疼变量的作用域.生存期.存储类别问题.什么静态.外部.寄存器.局部.全局搞得一头雾水.今天咱们就来梳理一下他们的变态关系(什么不得了的事情???) 1.变量的作用域 说 ...
- C语言基本类型和存储类别
基本类型 C语言的基本数据类型分为两大类:整数类型和浮点数类型.不同的种类提供了不同的范围和精度. 关键字 1. 创建基本数据类型要用到8个关键字:int.long.short.unsigned.ch ...
最新文章
- AdminLTE的使用
- php类属性命名驼峰还是下划线,PHP实现驼峰命名和下划线命名互转
- R语言使用apriori算法进行关联规则挖掘实战:关联规则概念、频繁项集、支持度(support)、置信度(confidence)、提升度(lift)、apriori算法
- python中if elif else流程图_python中的if、elif、else语法
- Java处理excel根据某列的值查询,并将结果显示在其他列中
- Xcode5下使用纯代码构建简单的HelloWorld程序
- 并发工具类纵览——建立起Java并发体系的大厦
- 青岛理工邀请赛(难受的一次经历)之显示屏(按着倍数放大数字)
- 已触发了一个断点 vs_VSCode源码分析-断点调试
- 业务自助分析怎么推?中梁集团办的这场BI建模大赛值得借鉴
- (87)FPGA面试题-同步FIFO与异步FIFO区别?异步FIFO代码设计
- python3对接聊天机器人API
- 读书笔记_中国期货市场量化交易(李尉)01
- 计算机控制面板程序可以设置鼠标,外设门诊:游戏中如何使用鼠标宏提升操作...
- 诺顿病毒库手工下载地址及英文补丁下载
- 脉冲计数器单片机c语言编程,基于单片机的光电计数器
- 智维专业工程师告诉你,哪款Kvaser多通道CAN总线分析仪性价比最高?
- 阿里粗排模型-cold
- 使用电脑过程中突然无法复制粘贴了
- 【STM32学习】(28)STM32实现光照度采集(标准库和HAL库实现)
热门文章
- 提升网站优化排名的前提是什么?
- mysql如何插入新的字段_Mysql 如何 得到新插入的字段ID
- pythonwith作用_老生常谈Python startswith()函数与endswith函数
- jpa 查找最后一条数据_查找数据的最后1条记录,你用了2小时,同事1分钟就搞定了...
- mac 思科 链路聚合_交换技术链路聚合配置
- c语言课设代写一般多少钱_结婚彩礼一般多少钱 2019彩礼会涨到多少钱
- for和foreach分析
- DGA域名可以是色情网站域名
- TFLearn 在给定模型精度时候提前终止训练
- suse linux通过iso文件安装gcc