在c語言中,每個變量和函數有兩個屬性:數據類型和數據的存儲類別。
C語言中局部變量和全局變量變量的存儲類別(static,extern,auto,register)
一. 從變量的作用域劃分變量(即從空間)角度來分
1.全局變量
2.局部變量
二. 從變量值存在的時間或存儲類別(即生存期)角度來分
2.1. 靜態存儲區
存放以下數據:代碼段(text)、只讀數據段(rodata)、讀寫數據段(rwdata)、未初始化數據段(bbs)
靜態存儲區存放全部的全局變量, 這些變量將在鏈接之后產生, 程序執行完畢就釋放, 程序執行的過程中它們占據固定的存儲單元, 而不會動態的進行分配和釋放
2.2. 動態存儲區
存放以下數據:函數形參、自動變量(未加static聲明的局部變量)、函數調用時的現場保護和返回地址
對以上這些數據,在函數開始調用時分配動態存儲空間,函數結束時釋放這些空間。
三. 從用戶內存空間角度分為三個部分
1. 程序區
2. 靜態存儲區
3. 動態存儲區
4. 從C程序運行時又可分為以下存儲區
1. 代碼段 (Code | Text)
代碼段由程序中執行的機器代碼組成。在C語言中,程序語句進行編譯后,形成機器代碼。在執行程序的過程中,CPU的程序計數器指向代碼段的每一條機器代碼,并由處理器依次運行。
2. 只讀數據段(ROData)
2.1 ROData介紹
只讀數據段是程序使用的一些不會被更改的數據,使用這些數據的方式類似查表式的操作,由于這些變量不需要更改,因此只需要放置在只讀存儲器中即可。只讀數據段由程序中所使用的數據產生,該部分數據的特點是在運行中不需要改變,因此編譯器會將該數據段放入只讀的部分中。C語言中的只讀全局變量,只讀局部變量,程序中使用的常量等會在編譯時被放入到只讀數據區。注意:定義全局變量const char a[100]={“ABCDEFG”};將生成大小為100個字節的只讀數據區,并使用“ABCDEFG”初始化。如果定義為:const char a[ ]={“ABCDEFG”};則根據字符串長度生成8個字節的只讀數據段(還有’’),所以在只讀數據段中,一般都需要做完全的初始化。
2.2 Example
3. 已初始化讀寫數據段(RW data)
3.1 RWData介紹
已初始化數據是在程序中聲明,并且具有初值的變量,這些變量需要占用存儲器的空間,在程序執行時它們需要位于可讀寫的內存區域內,并具有初值,以供程序運行時讀寫。全局變量全部存放在靜態存儲區,在程序開始執行時給全局變量分配存儲區,程序行完畢就釋放。在程序執行過程中它們占據固定的存儲單元,而不動態地進行分配和釋放;
全局變量
靜態(static) 局部變量
3.2 Example
4. 未初始化數據段(BSS)
4.1 BSS介紹
未初始化數據是在程序中聲明,但是沒有初始化的變量,這些變量在程序運行之前不需要占用存儲器的空間。
4.1 Example
5. 堆(heap)
5.1 堆空間介紹
堆內存只在程序運行時出現,一般由程序員分配和釋放。
在具有操作系統的情況下,如果程序沒有釋放,操作系統可能在程序(例如一個進程)結束后回收內存。
5.2 Example
6. 棧(stack)
6.1 棧空間介紹
棧內存只在程序運行時出現,在函數內部使用的變量、函數的參數以及返回值將使用棧空間, 棧空間由編譯器自動分配和釋放。棧空間是動態開辟與回收的。在函數調用過程中,如果函數調用的層次比較多,所需要的棧空間也逐漸加大對于參數的傳遞和返回值,如果使用較大的結構體,在使用的棧空間也會比較大。
6.2 棧區主要用于以下數據的存儲
函數內部的動態變量 函數的參數 函數的返回值
6.3 Example
5. 4種局部變量和全局變量的存儲類別(static,extern,auto,register)
5.1 Static
有時希望函數中的局部變量的值在函數調用結束后不消失而保留原值,這時就應該指定局部變量為“靜態局部變量”,用關鍵字static進行聲明。
對靜態局部變量的說明:
1)靜態局部變量屬于靜態存儲類別,在靜態存儲區內分配存儲單元。在程序整個運行期間都不釋放。而自動變量(即動態局部變量)屬于動態存儲類別,占動態存儲空間,函數調用結束后即釋放。
2)靜態局部變量在編譯時賦初值,即只賦初值一次;而對自動變量賦初值是在函數調用時進行,每調用一次函數重新給一次初值,相當于執行一次賦值語句。
3)如果在定義局部變量時不賦初值的話,則對靜態局部變量來說,編譯時自動賦初值0(對數值型變量)或空字符(對字符變量)。而對自動變量來說,如果不賦初值則它的值是一個不確定的值。
5.2 Extern
外部變量(即全局變量)是在函數的外部定義的,它的作用域為從變量定義處開始,到本程序文件的末尾。如果外部變量不在文件的開頭定義,其有效的作用范圍只限于定義處到文件終了。如果在定義點之前的函數想引用該外部變量,則應該在引用之前用關鍵字extern對該變量作“外部變量聲明”。表示該變量是一個已經定義的外部變量。有了此聲明,就可以從“聲明”處起,合法地使用該外部變量。
說明:
在本程序文件的最后1行定義了外部變量A,B,但由于外部變量定義的位置在函數main之后,因此本來在main函數中不能引用外部變量A,B。現在我們在main函數中用extern對A和B進行“外部變量聲明”,就可以從“聲明”處起,合法地使用該外部變量A和B。
5.3 Auto
函數中的局部變量,如不專門聲明為static存儲類別,都是動態地分配存儲空間的,數據存儲在動態存儲區中。函數中的形參和在函數中定義的變量(包括在復合語句中定義的變量),都屬此類,在調用該函數時系統會給它們分配存儲空間,在函數調用結束時就自動釋放這些存儲空間。這類局部變量稱為自動變量。自動變量用關鍵字auto作存儲類別的聲明。
a是形參,b,c是自動變量,對c賦初值3。執行完f函數后,自動釋放a,b,c所占的存儲單元。
關鍵字auto可以省略,auto不寫則隱含定為“自動存儲類別”,屬于動態存儲方式。占用棧空間
5.4 Register
為了提高效率,C語言允許將局部變量得值放在CPU中的寄存器中,這種變量叫“寄存器變量”,用關鍵字register作聲明。
說明:
1)只有局部自動變量和形式參數可以作為寄存器變量;
2)一個計算機系統中的寄存器數目有限,不能定義任意多個寄存器變量;
3)局部靜態變量不能定義為寄存器變量。
6. 總結
從變量的作用域(即從空間)角度來分,可以分為全局變量和局部變量。從變量值存在的作時間(即生存期)角度來分,可以分為靜態存儲方式和動態存儲方式。
代碼段、只讀數據段、讀寫數據段、未初始化數據段屬于靜態區域
靜態區域: 是指在程序運行期間分配固定的存儲空間的方式
堆和棧屬于動態區域
動態區域: 是在程序運行期間根據需要進行動態的分配存儲空間的方式。
代碼段、只讀數據段和讀寫數據段將在鏈接之后產生
未初始化數據段將在程序初始化的時候開辟而堆和棧將在程序的運行中分配和釋放。C語言程序分為映像和運行時兩種狀態。在編譯-連接后形成的映像中,將只包含代碼段(Text)、只讀數據段(RO Data)和讀寫數據段(RW Data)。在程序運行之前,將動態生成未初始化數據段(BSS) 在程序的運行時還將動態形成堆(Heap)區域和棧(Stack)區域。一般來說,在靜態的映像文件中,各個部分稱之為節(Section),而在運行時的各個部分稱之為段(Segment)。如果不詳細區分,可以統稱為段。C語言在編譯和連接后,將生成代碼段(Text)、只讀數據段(RO Data)和讀寫數據段(RW Data)。在運行時,除了以上三個區域外,還包括未初始化數據段(BSS)區域和堆(Heap)區域和棧(Stack)區域。
7. 一些實例
責任編輯人:CC
-
C語言
+關注
關注
180文章
7614瀏覽量
137433 -
C語言編程
+關注
關注
6文章
90瀏覽量
21135
原文標題:【精典C編程】C語言編程程序的內存分配總結
文章出處:【微信號:KY_QRS,微信公眾號:開源嵌入式】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論