C语言基础 — ( 变量的存储方式、生存期、内存机制 )

欢迎小伙伴的点评✨✨ 本篇章系列是对C语言的深度思考和总结、关于C语言内容会持续更新


前言

变量的作用域:每一个变量都有一个作用域,即它们在什么范围内有效。
一般为了叙述方便,把建立存储空间的声明称定义如(int a;),而把不需要建立存储空间的声明称声明 如(extern a;)。
本章节将会给大家带来数据在内存中的存储特性。
定义变量可能有3种情况:
(1) 在函数的开头定义;
(2) 在函数内的复合语句内定义;
(3) 在函数的外部定义。


一、局部变量和全局变量

1.1、局部变量

在一个函数内部定义的变量只在本函数范围内有效,也就是说只有在本函数内才能引用它们,在此函数以外是不能使用这些变量的。在复合语句内定义的变量只在本复合语句范围内有效,只有在本复合语句内才能引用它们。在该复合语句以外是不能使用这些变量的,以上这些称为 局部变量

说明:
(1)主函数中定义的变量(如m,n)也只在主函数中有效,并不因为在主函数中定义而在整个文件或程序中有效。主函数也不能使用其他函数中定义的变量。

(2)不同函数中可以使用同名的变量,它们代表不同的的对象,互不干扰。例如在F1函数中定义了变量b和c,倘若在F2函数中也定义变量b和c,它们在内存中占不同的单元,不会混淆。

(3)形式参数也是局部变量。例如在F1函数中的形参 a,也只在F1函数中有效。其他函数可以调用 F1函数,但不能直接引用F1函数的形参a (例如想在其他函数中输出a的值是不行的)。

(4)在一个函数内部,可以在复合语句中定义变量,这些变量只在本复合语句中有效,这种复合语句也称为**“分程序”“程序块”**
变量c只在复合语句(分程序)内有效,离开该复合语句该变量就无效,系统会把它占用的内存单元释放。

1.2、全局变量

前已介绍,程序的编译单位是源程序文件,一个源文件可以包含一个或若干个函数。在函数内定义的变量是局部变量,而在函数之外定义的变量称为外部变量,外部变量是全局变量(也称全程变量)。全局变量可以为本文件中其他函数所共用。它的有效范围为从定义变量的位置开始到本源文件结束。
注意: 在函数内定义的变量是局部变量,在函数外定义的变量是全局变量。
说明: 设置全局变量的作用是增加了函数间数据联系的渠道。由于同一文件中的所有函数都能引用全局变量的值,因此如果在一个函数中改变了全局变量的值,就能影响到其他函数中全局变量的值。相当于各个函数间有直接的传递通道。由于函数的调用只能带回一个函数返回值,因此有时可以利用全局变量来增加函数间的联系渠道,通过函数调用能得到一个以上的值。
但是,建议不在必要时不要使用全局变量,原因如下:
① 全局变量在程序的全部执行过程中都占用存储单元,而不是仅在需要时才开辟单元。
② 它使函数的通用性降低了,因为如果在函数中引用了全局变量,那么执行情况会受到有关的外部变量的影响,如果将一个函数移到另一个文件中,还要考虑把有关的外部变量及其值一起移过去。但是若该外部变量与其他文件的变量同名时,就会出现问题。这就降低了程序的可靠性和通用性。在程序设计中,在划分模块时要求模块的 内聚性 强、与其他模块的 耦合性 弱。即模块的功能要单一(不要把许多互不相干的功能放到一个模块中),与其他模块的相互影响要尽量少,而用全局变量是不符合这个原则的。一般要求把C程序中的函数做成一个相对的封闭体,除了可以通过 实参一形参 的渠道与外界发生联系外,没有其他渠道。这样的程序移植性好,可读性强。
③使用全局变量过多,会降低程序的清晰性,人们往往难以清楚地判断出每个瞬时各个外部变量的值。由于在各个函数执行时都可能改变外部变量的值,程序容易出错。因此,要限制使用全局变量。

注意: 如果在同一个源文件中,全局变量与局部变量同名,全局变量被局部变量屏蔽,即局部变量有效。

二、变量的存储方式和生存期

2.1、动态存储方式与静态存储方式

从变量的作用域(即从空间)的角度来观察,变量可以分为全局变量局部变量。
还可以从另一个角度,即从变量值存在的时间(即生存期)来观察。有的变量在程序运行的整个过程都是存在的,而有的变量则是在调用其所在的函数时才临时分配存储单元,而在函数调用结束后该存储单元就马上释放了,变量不存在了。也就是说,变量的存储有两种不同的方式:静态存储方式动态存储方式。静态存储方式是指在程序运行期间由系统分配固定的存储空间的方式,而动态存储方式则是在程序运行期间根据需要进行动态的分配存储空间的方式。
先看一下内存中的供用户使用的存储空间情况。这个存储空间可以分为3部分:
(1)程序区;
(2)静态存储区;
(3)动态存储区;
数据分别存放在静态存储区和动态存储区中。
全局变量全部存放在静态存储区中,在程序开始执行时给全局变量分配存储区,程序执行完毕就释放。在程序执行过程中它们占据固定的存储单元,而不是动态地进行分配和释放。
动态存储区中存放以下数据:
①函数形式参数。在调用函数时给形参分配存储空间。
②函数中定义的没有用关键字 static 声明的变量,即自动变量。
③函数调用时的现场保护和返回地址等。
对以上这些数据,在函数调用开始时分配动态存储空间,函数结束时释放这些空间。在程序执行过程中,这种分配和释放是动态的,如果在一个程序中两次调用同一函数,而在此函数中定义了局部变量,在两次调用时分配给这些局部变量的存储空间的地址可能是不相同的。
如果一个程序中包含若干个函数,每个函数中的局部变量的生存期并不等于整个程序的执行周期,它只是程序执行周期的一部分。
在程序执行过程中,先后调用各个函数,此时会动态地分配和释放存储空间。
在C语言中,每一个变量和函数都有两个属性:数据类型数据的存储类别。存储类别指的是数据在内存中存储的方式(如静态存储和动态存储)。
在定义和声明变量和函数时,一般应同时指定其数据类型和存储类别,也可以采用默认方式指定(即如果用户不指定,系统会隐含地指定为某一种存储类别)。
C的存储类别包括4种:自动的(auto)、静态的(statis)、寄存器的(register)、外部的(extern)。根据变量的存储类别,可以知道变量的作用域和生存期。下面分别作介绍。

2.2、局部变量的存储类别

(1) 自动变量(auto 变量)
函数中的局部变量,如果不专门声明为 static(静态)存储类别,都是动态地分配存储空间的,数据存储在动态存储区中。函数中的形参和在函数中定义的局部变量(包括在复合语句中定义的局部变量),都属于此类。在调用该函数时,系统会给这些变量分配存储空间,在函数调用结束时就自动释放这些存储空间。因此这类局部变量称为自动变量。自动变量用关键字 auto 作存储类别的声明。
实际上,关键字 auto 可以省略,不写 auto 则隐含指定为“自动存储类别”,它属于动态存储方式。程序中大多数变量属于自动变量。在函数中定义的变量都没有声明为 auto,其实都隐含指定为自动变量。例如,在函数体中:
int ,c=3;

auto int b,c=3;
等价。
(2) 静态局部变量(static 局部变量)
有时希望函数中的局部变量的值在函数调用结束后不消失而继续保留原值,即其占用的存储单元不释放,在下一次再调用该函数时,该变量已有值(就是上一次函数调用结束时的值)。这时就应该指定该局部变量为“静态局部变量”,用关键字 static 进行声明。

说明:
(1)静态局部变量属于静态存储类别,在静态存储区内分配存储单元。在程序整个运行期间都不释放。而自动变量(即动态局部变量)属于动态存储类别,分配在动态存储区空间而不在静态存储区空间,函数调用结束后即释放。
(2)对静态局部变量是在编译时赋初值的,即只赋初值一次,在程序运行时它已有初值。以后每次调用函数时不再重新赋初值而只是保留上次函数调用结束时的值。而对自动变量赋初值,不是在编译时进行的,而是在函数调用时进行的,每调用一次函数重新给一次初值,相当于执行一次赋值语句。
(3)如果在定义局部变量时不赋初值的话,则对静态局部变量来说,编译时自动赋初值0(对数值型变量)或空字符\0’(对字符变量)。而对自动变量来说,它的值是一个不确定的值。这是由于每次函数调用结束后存储单元已释放,下次调用时又重新另分配存储单元,而所分配的单元中的内容是不可知的。
(4)虽然静态局部变量在函数调用结束后仍然存在,但其他函数是不能引用它的。因为它是局部变量,只能被本函数引用,而不能被其他函数引用。

(3) 寄存器变量(register变量)
一般情况下,变量(包括静态存储方式和动态存储方式)的值是存放在内存中的。当程序中用到哪一个变量的值时,由控制器发出指令将内存中该变量的值送到运算器中。经过运算器进行运算,如果需要存数,再从运算器将数据送到内存存放。
如果有一些变量使用频繁(例如,在一个函数中执行10000次循环,每次循环中都要引用某局部变量),则为存取变量的值要花费不少时间。为提高执行效率,允许将局部变量的值放在 CPU 中的寄存器中,需要用时直接从寄存器取出参加运算,不必再到内存中去存取。由于对寄存器的存取速度远高于对内存的存取速度,因此这样做可以提高执行效率。这种变量叫做寄存器变量,用关键字register作声明。如 register int f; //定义f为寄存器变量
由于现在的计算机的速度愈来愈快,性能愈来愈高,优化的编译系统能够识别使用频繁的变量,从而自动地将这些变量放在寄存器中,而不需要程序设计者指定。因此,现在实际上用register声明变量的必要性不大。
注意: 3种局部变量的存储位置是不同的:自动变量存储在动态存储区;静态局部变量存储在静态存储区;寄存器存储在CPU中 的寄存器中。

2.3、全局变量的存储类别

全局变量都是存放在静态存储区中的。因此它们的生存期是固定的,存在于程序的整个运行过程。但是,对全局变量来说,还有一个问题尚待解决,就是它的作用域究竟从什么位置起,到什么位置止。作用域是包括整个文件范围还是文件中的一部分范围?是在一个文件中有效还是在程序的所有文件中都有效?这就需要指定不同的存储类别。
一般来说,外部变量是在函数的外部定义的全局变量,它的作用域是从变量的定义处开始,到本程序文件的末尾。在此作用域内,全局变量可以为程序中各个函数所引用。但有时程序设计人员希望能扩展外部变量的作用域。有以下几种情况。

(1) 在一个文件内扩展外部变量的作用域
如果外部变量不在文件的开头定义,其有效的作用范围只限于定义处到文件结束。在定义点之前的函数不能引用该外部变量。如果由于某种考虑,在定义点之前的函数需要引用该外部变量,则应该在引用之前用关键字 extern 对该变量作**“外部变量声明”**,表示把该外部变量的作用域扩展到此位置。有了此声明,就可以从 声明 处起,合法地使用该外部变量。例如:

extern int A , B , C ; //把外部变量A,B,C的作用域扩展到从此处开始
int A , B , C ; //定义外部变量A,B,C

注意: 提倡将外部变量的定义放在引用它的所有函数之前,这样可以避免在函数中多加一个extern 声明。

**(2) 将外部变量的作用域扩展到其他文件 **
一个C程序可以由一个或多个源程序文件组成。如果程序只由一个源文件组成,使用外部变量的方法前面已经介绍。如果程序由多个源程序文件组成,那么在一个文件中想引用另一个文件中已定义的外部变量,有什么办法呢?
如果一个程序包含两个文件,在两个文件中都要用到同一个外部变量 A,不能分别在两个文件中各自定义一个外部变量 A,否则在进行程序的连接时会出现“重复定义”的错误。正确的做法是:在任一个文件中定义外部变量 A,而在另一文件中用 extern 对A作 外部变量声明 ,即 extern A; 。在编译和连接时,系统会由此知道 A有 外部链接 ,可以从别处找到已定义的外部变量 A,并将在另一文件中定义的外部变量A的作用域扩展到本文件,在本文件中可以合法地引用外部变量 A 。
例如:
文件file_1.c:
int A; //定义外部变量
int main()
{

return 0;
}
文件file_2.c:
extern A; //把file_1.c文件中已定义的外部变量的作用域扩展到本文件。

说明: 用这种方法扩展全局变量的作用域应十分慎重,因为在执行一个文件中的操作时,可能会改变全局的值,会影响到另一个文件中全局变量的值,从而影响该文件中函数执行结果。
extern 既可以用来扩展外部变量在本文件中的作用域,又可以使外部变量的作用域从一个文件扩展到程序中的其他文件,那么系统怎么区别处理呢?实际上,在编译时遇到 extern 时,先在本文件中找外部变量的定义,如果找到,就在本文件中扩展作用域;如果找不到,就在连接时从其他文件中找外部变量的定义。如果从其他文件中找到了,就将作用域扩展到本文件;如果再找不到,就按出错处理。

(3) 将外部变量的作用域限制在本文件中

有时在程序设计中希望某些外部变量只限于被本文件引用,而不能被其他文件引用。
这时可以在定义外部变量时加一个static声明。
例如:
file_1.c
static int A; //全局变量只限于被本文本引用
int main()
{

return 0;
}

file_2.c
extern A; //出错
void FUN()
{

}

在 filel_1. c 中定义了一个全局变量A,但它用了static 声明,把变量A的作用域限制在本文件范围内,虽然在 file_2 中用了 extern A;
,但仍然不能使用 filel. c中的全局变量A。这种加上 static 声明、只能用于本文件的外部变量称为静态外部变量。在程序设计中,常由若干人分别完成各个模块,各人可以独立地在其设计的文件中使用相同的外部变量名而互不相干。只须在每个文件中定义外部变量时加上 static 即可。这就为程序的模块化、通用性提供方便。如果已确认其他文件不需要引用本文件的外部变量,就可以对本文件中的外部变量都加上 static,成为静态外部变量,以免被其他文件误用。这就相当于把本文件的外部变量对外界 屏蔽 起来,从其他文件的角度看,这个静态外部变量是 看不见,不能用 的。至于在各文件中在函数内定义的局部变量,本来就不能被函数外引用,更不能被其他文件引用,因此是安全的。

说明: 不要误认为对外部变量加 static 声明后才采取静态存储方式(存放在静态存储区中),而不加 static 的是采取动态存储(存放在动态存储区)。声明局部变量的存储类型和声明全局变量的存储类型的含义是不同的。对于局部变量来说,声明存储类型的作用是指定变量存储的区域(静态存储区或动态存储区)以及由此产生的生存期的问题,而对于全局变量来说,由于都是在编译时分配内存的,都存放在静态存储区,声明存储类型的作用是变量作用域的扩展问题。

注意: 用auto,register 和 static 声明变量时,是在定义变量的基础上加上这些关键字,而不能单独使用。下面的用法不对:
int a;//先定义整型变量 a
static a //企图再将变量 a 声明为静态变量
编译时会被认为**“重新定义”**。

三、存储类别小结

从以上可知,对一个数据的定义,需要指定两种属性:数据类型存储类别,分别使用两个关键字。例如:
static int a; //静态局部整型变量或静态外部整型变量
auto char c; //自动变量,在函数内定义
register int d; //寄存器变量,在函数内定义
此外,可以用 extern 声明已定义的外部变量,例如:
extern b; //将已定义的外部变量b的作用域扩展至此下面从不同角度做些归纳:
下面从不同角度做些归纳:
(1) 从作用域角度分,有局部变量和全局变量。它们采用的存储类别如下:

局部变量
局部变量
局部变量
局部变量
全局变量
全局变量
按作用域角度分
自动变量:即动态局部变量:离开函数-值就消失
静态局部变量:离开函数-值仍保留
寄存器变量:离开函数-值就消失
形式参数可以定义为自动变量或寄存器变量
静态外部变量:只限本文件引用
外部变量:即非静态的外部变量允许其他文件引用

(2) 从变量存在的时间(生命期)来区分,有动态存储和静态存储两种类型。静态存储是程序整个运行时间都存在,而动态存储则是在调用函数时临时分配单元。

动态存储
动态存储
动态存储
静态存储
静态存储
静态存储
按变量的生存期分
自动变量:本函数内有效
寄存器变量:本函数内有效
形式参数:本函数内有效
静态局部变量:函数内有效
静态外部变量:本文件内有效
外部变量:用extern 声明后-其他文件可以引用

(3) 从变量值存放的位置来区分,可分为:

内存中静态存储区
内存中静态存储区
内存中静态存储区
内存中动态存储区
CPU中的寄存器
按变量值存放位置分
静态局部变量
静态外部变量:函数外部静态变量
外部变量:可以为其他文件引用
自动变量和形式参数
寄存器变量

(4)关于作用域和生存期的概念。从前面叙述可以知道,对一个变量的属性可以从两个方面分析,一是变量的作用域,一是变量值存在时间的长短,即生存期。前者是从空间的角度,后者是从时间的角度。二者有联系但不是同一回事。
如果一个变量在某个文件或函数范围内是有效的,就称该范围为该变量的作用域,在此作用域内可以引用该变量,这种性质也可以称为变量的可见性
如果一个变量值在某一时刻是存在的,则认为这一时刻属于该变量的生存期,或称该变量在此时此刻存在

(5) static 对局部变量和全局变量的作用不同。对局部变量来说,它使变量由动态存储方式改变为静态存储方式。而对全局变量来说,它使变量局部化(局部于本文件),但仍为静态存储方式。从作用域角度看,凡有 static 声明的,其作用域都是局限的,或者局限于本函数内(静态局部变量),或者局限于本文件内(静态外部变量)。

猜你喜欢

转载自blog.csdn.net/weixin_44759598/article/details/128591161
今日推荐