全域性變數與靜態全域性變數的異同

2021-09-28 15:52:32 字數 2335 閱讀 9661

參考:

1、生存週期: 變數從定義到銷毀的時間範圍。存放在全域性資料區的變數的生存週期存在於整個程式執行期間,而存放在棧中的資料則隨著函式等的作用域結束導致出棧而銷毀,除了靜態變數之外的區域性變數都存放於棧中。

2、作用域: 變數的可見**域(塊作用域,函式作用域,類作用域,程式全域性作用域)。

3、static變數是指靜態的變數,不管是在全域性還是區域性宣告的static變數都存放於程式的全域性變數區域,所以它的生命週期是從程式開始到程式結束。但是static變數的作用域並不等同於它的生存週期,它的作用域決定於它被定義的位置。可以認為static變數的作用域<=生存週期。

舉乙個區域性宣告的例子。在函式test中宣告靜態變數i:

void test()

區域性變數m存放在棧中,當test函式結束,m將被銷毀;靜態變數i不存放在棧中,而是存放於程式的全域性變數區域,因此隨著函式test的結束,它並不隨著出棧操作而被銷毀,它的生存週期存在於程式的整個執行期;然而m和i的作用域都僅存在於test函式中它們的定義之後,即test呼叫結束之後,m和i就不再可用,但是i仍存在於記憶體之中。

再舉乙個全域性宣告的例子。在檔案a 中定義靜態變數j:

int n=3; //預設為extern

static int j=5; //宣告為static

全域性變數和靜態變數j都存放於程式的全域性資料區域,它們的生存週期都是程式的整個執行期,但是n的作用域為全域性作用域,可以通過extern在其他檔案中使用,而j只能在檔案a中使用,例如在檔案b中:

extern int n; //ok

extern int j; //error: j在檔案b中不可見

int a=n;//ok:但這裡有個初始化先後的問題,具體參見參考一

int b=j;//error

也就是說,在宣告全域性的static變數時,static沒有改變它的生存週期,也即儲存位置(因為全域性變數本來就儲存在全域性資料域),而是將變數的作用域限制在當前檔案中。

中興通訊2012校招筆試題的一道問答題:

static全域性變數與普通的全域性變數有什麼區別 ?

全域性變數(外部變數)的說明之前再冠以static 就構成了靜態的全域性變數。

全域性變數本身就是靜態儲存方式, 靜態全域性變數當然也是靜態儲存方式。 這兩者在儲存方式上並無不同。

這兩者的區別在於非靜態全域性變數的作用域是整個源程式, 當乙個源程式由多個原始檔組成時,非靜態的全域性變數在各個原始檔中都是有效的。 而靜態全域性變數則限制了其作用域, 即只在定義該變數的原始檔內有效, 在同一源程式的其它原始檔中不能使用它。由於靜態全域性變數的作用域侷限於乙個原始檔內,只能為該原始檔內的函式公用,因此可以避免在其它原始檔中引起錯誤。

static全域性變數只初使化一次,防止在其他檔案單元中被引用;

static區域性變數和普通區域性變數有什麼區別 ?

把區域性變數改變為靜態變數後是改變了它的儲存方式即改變了它的生存期。把全域性變數改變為靜態變數後是改變了它的作用域,限制了它的使用範圍。

static區域性變數只被初始化一次,下一次依據上一次結果值;

static函式與普通函式有什麼區別?

static函式與普通函式作用域不同,僅在本檔案。只在當前原始檔中使用的函式應該說明為內部函式(static修飾的函式),內部函式應該在當前原始檔中說明和定義。對於可在當前原始檔以外使用的函式,應該在乙個標頭檔案中說明,要使用這些函式的原始檔要包含這個標頭檔案.

static函式在記憶體中只有乙份,普通函式在每個被呼叫中維持乙份拷貝

p.s.

c程式一直由下列部分組成:

1)正文段(**段)——cpu執行的機器指令部分;乙個程式只有乙個副本;唯讀,防止程式由於意外事故而修改自身指令;

2)初始化資料段(資料段)——在程式中所有賦了初值的全域性變數,存放在這裡。

3)非初始化資料段(bss段)——在程式中沒有初始化的全域性變數;核心將此段初始化為0。(這就是為什麼全域性內建型別變數會初始化,而區域性變數就為未初始化的未知值)

4)棧——增長方向:自頂向下增長;自動變數以及每次函式呼叫時所需要儲存的資訊(返回位址;環境資訊)。

5)堆——動態儲存分。

程式在記憶體中申請了**段,全域性資料段(初始化和未初始化),棧和堆:程式**放於**段,全域性變數和靜態變數存放在全域性資料段中,一直存在直到程式結束,而區域性變數都放於臨時的棧中,隨著作用域的結束隨著出棧操作而銷毀。malloc和new出來的記憶體不屬於上面提到的程式申請的記憶體中,而是在系統中申請到的記憶體,所以如果在程式中沒有明確free和delete的話,程式結束後該記憶體仍不會被釋放,造成記憶體洩漏。例如:

int test()

則test函式結束後,p指標本身被銷毀,因為它是存在於函式棧中的區域性變數,而p指向的int整型內容仍存在於內容堆中,沒有被釋放,造成記憶體洩漏。

詳細記憶體使用和參考

全域性變數和靜態全域性變數

全域性變數和區域性變數是從變數的作用域的角度劃分。靜態變數和動態變數是從變數的記憶體分配的角度劃分。全域性變數本身就是靜態儲存方式,靜態全域性變數當然也是靜態儲存方式。這兩者在儲存方式上並無不同,區別在於非靜態全域性變數的作用域是整個源程式,當乙個源程式由多個原始檔組成時,非靜態的全域性變數在各個原...

c 全域性變數 靜態全域性變數

全域性變數是靜態儲存方式,靜態全域性變數也是靜態儲存方式,這兩者在儲存方式上並無不同。區別 雖在於非靜態全域性變數的作用域是整個源程式,當乙個源程式由多個原始檔組成時,靜態全域性變數在各個原始檔中都是有效的。靜態區域性變數則限制了其作用域,只在定義該變數的原始檔內有效,在同一源程式的其它原始檔中不能...

C 靜態全域性變數與普通全域性變數

一 儲存區域 1 棧 由編譯器在需要的時候分配,在不需要的時候自動清除的變數的儲存區域。通常儲存區域性變數,函式引數。2 堆 由new分配的記憶體塊,需要手動釋放。如果程式設計師沒有手動釋放,在程式執行結束後,作業系統自動 3 自由儲存區 由malloc等分配的記憶體塊,與堆相似,用free來釋放。...