全局變量&局部變量及其作用域&作用週期

原鏈接:點擊打開鏈接

C語言中所有變量都有自己的作用域,申明變量的類型不同,其作用域也不同。C語言中的變量,按照作用域的範圍可分爲兩種, 即局部變量和全局變量。 一、局部變量

局部變量也稱爲內部變量。局部變量是在函數內作定義說明的。其作用域僅限於函數內, 離開該函數後再使用這種變量是非法的。

例如:

int f1(int a) /*函數f1*/ {     int b,c

……}

int f2(int x) /*函數f2*/ {     int y,z;

……}

f1內定義了三個變量,a爲形參,b,c爲一般變量。在 f1的範圍內a,b,c有效,或者說a,b,c變量的作用域限於f1內。

f2內定義了三個變量,x爲形參,y,z爲一般變量。在 f2的範圍內x,y,z有效,或者說x,y,z變量的作用域限於f2內。

關於局部變量的作用域還要說明以下幾點:

         主函數中定義的變量也只能在主函數中使用,不能在其它函數中使用。同時,主函數中也不能使用其它函數中定義的變量。因爲主函數也是一個函數,它與其它函數是平行關係。

         形參變量是屬於被調函數的局部變量,實參變量是屬於主調函數的局部變量。

         允許在不同的函數中使用相同的變量名,它們代表不同的對象,分配不同的單元,互不干擾,也不會發生混淆。雖然允許在不同的函數中使用相同的變量名,但是爲了使程序明瞭易懂,不提倡在不同的函數中使用相同的變量名。

二、全局變量

int a,b; /*外部變量*/ void f1() /*函數f1*/ { …… }

float x,y; /*外部變量*/ int fz() /*函數fz*/ { …… }

全局變量也稱爲外部變量,它是在函數外部定義的變量。 它不屬於哪一個函數,它屬於一個源程序文件。其作用域是整個源程序。在函數中使用全局變量,一般應作全局變量說明。 只有在函數內經過說明的全局變量才能使用。全局變量的說明符爲extern 但在一個函數之前定義的全局變量,在該函數內使用可不再加以說明。 例如:   從上例可以看出abx都是在函數外部定義的外部變量,都是全局變量。  對於全局變量還有以下幾點說明:

         對於局部變量的定義和說明,可以不加區分。而對於外部變量則不然,外部變量的定義和外部變量的說明並不是一回事。外部變量定義必須在所有的函數之外,且只能定義一次。其一般形式爲: [extern] 類型說明符 變量名,變量名… 其中方括號內的extern可以省去不寫。    例如: int a,b;    等效於:  extern int a,b;    而外部變量說明出現在要使用該外部變量的各個函數內,在整個程序內,可能出現多次,外部變量說明的一般形式爲: extern 類型說明符 變量名,變量名, 外部變量在定義時就已分配了內存單元,外部變量定義可作初始賦值,外部變量說明不能再賦初始值, 只是表明在函數內要使用某外部變量。

         外部變量可加強函數模塊之間的數據聯繫,但是又使函數要依賴這些變量,因而使得函數的獨立性降低。從模塊化程序設計的觀點來看這是不利的, 因此在不必要時儘量不要使用全局變量。

         在同一源文件中,允許全局變量和局部變量同名。在局部變量的作用域內,全局變量不起作用。 int vs(int l,int w) { extern int h; int v; v=l*w*h; return v; } main() { extern int w,h; int l=5; printf("v=%d",vs(l,w)); } int l=3,w=4,h=5;    本例程序中,外部變量在最後定義, 因此在前面函數中對要用的外部變量必須進行說明。外部變量lwvs函數的形參lw同名。外部變量都作了初始賦值,mian函數中也對l作了初始化賦值。執行程序時,在printf語句中調用vs函數,實參l的值應爲main中定義的l值,等於5,外部變量lmain內不起作用;實參w的值爲外部變量w的值爲4,進入vs後這兩個值傳送給形參lwvs函數中使用的爲外部變量,其值爲5,因此v的計算結果爲100,返回主函數後輸出。

變量的存儲類型決定了各種變量的作用域不同。所謂存儲類型是指變量佔用內存空間的方式,也稱爲存儲方式。變量的存儲方式可分爲靜態存儲動態存儲兩種。    靜態存儲變量通常是在變量定義時就分定存儲單元並一直保持不變, 直至整個程序結束。動態存儲變量是在程序執行過程中,使用它時才分配存儲單元, 使用完畢立即釋放。 典型的例子是函數的形式參數,在函數定義時並不給形參分配存儲單元,只是在函數被調用時,才予以分配, 調用函數完畢立即釋放。如果一個函數被多次調用,則反覆地分配、 釋放形參變量的存儲單元。從以上分析可知, 靜態存儲變量是一直存在的, 而動態存儲變量則時而存在時而消失。我們又把這種由於變量存儲方式不同而產生的特性稱變量的生存期。 生存期表示了變量存在的時間。 生存期和作用域是從時間和空間這兩個不同的角度來描述變量的特性,這兩者既有聯繫,又有區別。 一個變量究竟屬於哪一種存儲方式, 並不能僅從其作用域來判斷,還應有明確的存儲類型說明。

C語言中,對變量的存儲類型說明有以下四種:

auto     自動變量   register    寄存器變量   extern    外部變量    static    靜態變量    自動變量和寄存器變量屬於動態存儲方式,外部變量和靜態變量屬於靜態存儲方式。在介紹了變量的存儲類型之後, 可以知道對一個變量的說明不僅應說明其數據類型,還應說明其存儲類型。 因此變量說明的完整形式應爲: 存儲類型說明符 數據類型說明符 變量名,變量名 例如:    static int a,b;           說明a,b爲靜態類型變量    auto char c1,c2;          說明c1,c2爲自動字符變量    static int a[5]={1,2,3,4,5};    說明a爲靜整型數組    extern int x,y;           說明x,y爲外部整型變量    下面分別介紹以上四種存儲類型:    一、自動變量的類型說明符爲auto    這種存儲類型是C語言程序中使用最廣泛的一種類型。C語言規定, 函數內凡未加存儲類型說明的變量均視爲自動變量, 也就是說自動變量可省去說明符auto 在前面各章的程序中所定義的變量凡未加存儲類型說明符的都是自動變量。例如: { int i,j,k; char c; …… }等價於: { auto int i,j,k; auto char c; …… }    自動變量具有以下特點:    1. 自動變量的作用域僅限於定義該變量的個體內。在函數中定義的自動變量,只在該函數內有效。在複合語句中定義的自動變量只在該複合語句中有效。 例如: int kv(int a) { auto int x,y; { auto char c; } /*c的作用域*/ …… } /*a,x,y的作用域*/    2. 自動變量屬於動態存儲方式,只有在使用它,即定義該變量的函數被調用時纔給它分配存儲單元,開始它的生存期。函數調用結束,釋放存儲單元,結束生存期。因此函數調用結束之後,自動變量的值不能保留。在複合語句中定義的自動變量,在退出複合語句後也不能再使用,否則將引起錯誤。例如以下程序: main() { auto int a,s,p; printf("/ninput a number:/n"); scanf("%d",&a); if(a>0){ s=a+a; p=a*a; } printf("s=%d p=%d/n",s,p); } { auto int a; printf("/ninput a number:/n"); scanf("%d",&a); if(a>0){ auto int s,p; s=a+a; p=a*a; } printf("s=%d p=%d/n",s,p); }    s,p是在複合語句內定義的自動變量,只能在該複合語句內有效。而程序的第9行卻是退出複合語句之後用printf語句輸出s,p的值,這顯然會引起錯誤。    3. 由於自動變量的作用域和生存期都侷限於定義它的個體內函數或複合語句內) 因此不同的個體中允許使用同名的變量而不會混淆。 即使在函數內定義的自動變量也可與該函數內部的複合語句中定義的自動變量同名。例5.14表明了這種情況。 main() { auto int a,s=100,p=100; printf("/ninput a number:/n"); scanf("%d",&a); if(a>0) { auto int s,p; s=a+a; p=a*a; printf("s=%d p=%d/n",s,p); } printf("s=%d p=%d/n",s,p); }    本程序在main函數中和複合語句內兩次定義了變量s,p爲自動變量。按照C語言的規定,在複合語句內,應由複合語句中定義的s,p起作用,故s的值應爲a+ ap的值爲a*a。退出複合語句後的s,p 應爲main所定義的s,p,其值在初始化時給定,均爲100。從輸出結果可以分析出兩個s和兩個p雖變量名相同, 但卻是兩個不同的變量。    4. 對構造類型的自動變量如數組等,不可作初始化賦值。    二、外部變量的類型說明符爲extern    在前面介紹全局變量時已介紹過外部變量。這裏再補充說明外部變量的幾個特點:    1. 外部變量和全局變量是對同一類變量的兩種不同角度的提法。全局變量是從它的作用域提出的,外部變量從它的存儲方式提出的,表示了它的生存期。    2. 當一個源程序由若干個源文件組成時, 在一個源文件中定義的外部變量在其它的源文件中也有效。例如有一個源程序由源文件F1.CF2.C組成: F1.C int a,b; /*外部變量定義*/ char c; /*外部變量定義*/ main() { …… }    F2.C extern int a,b; /*外部變量說明*/ extern char c; /*外部變量說明*/ func (int x,y) { …… }    F1.CF2.C兩個文件中都要使用a,b,c三個變量。在F1.C文件中把a,b,c都定義爲外部變量。在F2.C文件中用extern把三個變量說明爲外部變量,表示這些變量已在其它文件中定義,並把這些變量的類型和變量名,編譯系統不再爲它們分配內存空間。 對構造類型的外部變量, 如數組等可以在說明時作初始化賦值,若不賦初值,則系統自動定義它們的初值爲0 靜態變量    靜態變量的類型說明符是static 靜態變量當然是屬於靜態存儲方式,但是屬於靜態存儲方式的量不一定就是靜態變量, 例如外部變量雖屬於靜態存儲方式,但不一定是靜態變量,必須由 static加以定義後才能成爲靜態外部變量,或稱靜態全局變量。 對於自動變量,前面已經介紹它屬於動態存儲方式。 但是也可以用static定義它爲靜態自動變量,或稱靜態局部變量,從而成爲靜態存儲方式。 由此看來, 一個變量可由static進行再說明,並改變其原有的存儲方式。    1. 靜態局部變量    在局部變量的說明前再加上static說明符就構成靜態局部變量。    例如: static int a,b; static float array[5]={1,2,3,4,5}    靜態局部變量屬於靜態存儲方式,它具有以下特點:    (1)靜態局部變量在函數內定義,但不象自動變量那樣,當調用時就存在,退出函數時就消失。靜態局部變量始終存在着,也就是說它的生存期爲整個源程序。    (2)靜態局部變量的生存期雖然爲整個源程序,但是其作用域仍與自動變量相同,即只能在定義該變量的函數內使用該變量。退出該函數後, 儘管該變量還繼續存在,但不能使用它。    (3)允許對構造類靜態局部量賦初值。在數組一章中,介紹數組初始化時已作過說明。若未賦以初值,則由系統自動賦以0值。    (4)對基本類型的靜態局部變量若在說明時未賦以初值,則系統自動賦予0值。而對自動變量不賦初值,則其值是不定的。 根據靜態局部變量的特點, 可以看出它是一種生存期爲整個源程序的量。雖然離開定義它的函數後不能使用,但如再次調用定義它的函數時,它又可繼續使用, 而且保存了前次被調用後留下的值。 因此,當多次調用一個函數且要求在調用之間保留某些變量的值時,可考慮採用靜態局部變量。雖然用全局變量也可以達到上述目的,但全局變量有時會造成意外的副作用,因此仍以採用局部靜態變量爲宜。    [5.15] main() { int i; void f(); /*函數說明*/ for(i=1;i<=5;i++) f(); /*函數調用*/ } void f() /*函數定義*/ { auto int j=0; ++j; printf("%d/n",j); }    程序中定義了函數f,其中的變量說明爲自動變量並賦予初始值爲0。當main中多次調用f時,j均賦初值爲0,故每次輸出值均爲1。現在把j改爲靜態局部變量,程序如下: main() { int i; void f(); for (i=1;i<=5;i++) f(); } void f() { static int j=0; ++j; printf("%d/n",j); } void f() { static int j=0; ++j; printf("%d/n",j); }    由於j爲靜態變量,能在每次調用後保留其值並在下一次調用時繼續使用,所以輸出值成爲累加的結果。讀者可自行分析其執行過程。    2.靜態全局變量    全局變量(外部變量)的說明之前再冠以static 就構成了靜態的全局變量。全局變量本身就是靜態存儲方式, 靜態全局變量當然也是靜態存儲方式。 這兩者在存儲方式上並無不同。這兩者的區別雖在於非靜態全局變量的作用域是整個源程序, 當一個源程序由多個源文件組成時,非靜態的全局變量在各個源文件中都是有效的。 而靜態全局變量則限制了其作用域, 即只在定義該變量的源文件內有效, 在同一源程序的其它源文件中不能使用它。由於靜態全局變量的作用域侷限於一個源文件內,只能爲該源文件內的函數公用, 因此可以避免在其它源文件中引起錯誤。從以上分析可以看出, 把局部變量改變爲靜態變量後是改變了它的存儲方式即改變了它的生存期。把全局變量改變爲靜態變量後是改變了它的作用域, 限制了它的使用範圍。因此static 這個說明符在不同的地方所起的作用是不同的。應予以注意。    四、寄存器變量    上述各類變量都存放在存儲器內, 因此當對一個變量頻繁讀寫時,必須要反覆訪問內存儲器,從而花費大量的存取時間。 爲此,C語言提供了另一種變量,即寄存器變量。這種變量存放在CPU的寄存器中,使用時,不需要訪問內存,而直接從寄存器中讀寫, 這樣可提高效率。寄存器變量的說明符是register 對於循環次數較多的循環控制變量及循環體內反覆使用的變量均可定義爲寄存器變量。 ∑200i=1imain() { register i,s=0; for(i=1;i<=200;i++) s=s+i; printf("s=%d/n",s); }    本程序循環200次,is都將頻繁使用,因此可定義爲寄存器變量。對寄存器變量還要說明以下幾點:    1. 只有局部自動變量和形式參數纔可以定義爲寄存器變量。因爲寄存器變量屬於動態存儲方式。凡需要採用靜態存儲方式的量不能定義爲寄存器變量。    2. Turbo CMS C等微機上使用的C語言中, 實際上是把寄存器變量當成自動變量處理的。因此速度並不能提高。 而在程序中允許使用寄存器變量只是爲了與標準C保持一致。3. 即使能真正使用寄存器變量的機器,由於CPU 中寄存器的個數是有限的,因此使用寄存器變量的個數也是有限的。 

——————————————————————————————————————————

原鏈接:點擊打開鏈接
靜態變量,就是在定義的時候,有static 修飾的變量,形式爲
static TYPE var_name = init_value;
而動態變量,形式爲
TYPE var_name = init_value;
即沒有static 修飾。其中的=init_value均可省略。
區分定義在函數外的全局變量,和函數內的局部變量,作用域,生命週期,及無顯式初始化時的初始值,均有區別。
1 動態全局變量:
作用域爲整個項目,即最終編譯成可執行文件的所有文件中均可以使用動態全局變量。
生命週期爲從程序運行到程序退出,即貫穿整個運行時間。
無顯式初始化時默認初始化值爲0。

2 靜態全局變量
作用域爲當前文件,從定義/聲明位置到文件結尾。
生命週期爲從程序運行到程序退出,即貫穿整個運行時間。
無顯式初始化時默認初始化值爲0。

3 動態局部變量
作用域爲當前函數,從定義位置,到其所在的{}的結束位置。
生命週期爲從函數調用到函數退出。
無顯式初始化時默認初始化值爲隨機值。

4 靜態局部變量
作用域爲當前函數,從定義位置,到其所在的{}的結束位置。
生命週期爲從程序運行到程序退出,即貫穿整個運行時間,當下次函數調用時,靜態局部變量不會被再次初始化,而是沿用上次函數退出時的值。
無顯式初始化時默認初始化值爲0。

#include<stdio.h>
int fun3(int x)
{
	static int a=3;
	a+=x;
	return(a);
}
main()
{	
	int k=2, m=1, n;
	n=fun3(k);
	n=fun3(m);
	printf("%d\n",n);            //6
}
#include<stdio.h>
int fun(int x[],int n)
{
	static int sum=0,i;
	for(i=0;i<n;i++) sum+=x[i];
	return sum;
}
main()
{
	int a[]={1,2,3,4,5},b[]={6,7,8,9,},s=0;
	s=fun(a,5)+fun(b,4);
	printf("%d\n",s);   //60
}

#include<stdio.h>
int a,b;
void fun()
{
	a=100;
	b=200;
}

main()
{
	int a=5,b=7;
	fun();
	printf("%d,%d\n",a,b);        //5,7
}

#include<stdio.h>
void fun(){
	static int a=0;
	a+=2;
	printf("%d",a);
}

main(){
	int cc;
	for(cc=1;cc<4;cc++) fun();    //246
	printf("\n");
}

以下敘述中正確的是___A___。
A、局部變量說明爲static 存儲類,其生存期將得到延長
B、全局變量說明爲static 存儲類,其作用域將被擴大
C、任何存儲類的變量在未賦初值時,其值都是不確定的

D、形參可以使用的存儲類說明符與局部變量完全相同



發佈了28 篇原創文章 · 獲贊 7 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章