通過(guò)一個(gè)小例子來(lái)簡(jiǎn)單理解C語(yǔ)言中的內(nèi)存空間管理
對(duì)于一個(gè)C語(yǔ)言程序而言,內(nèi)存空間主要由五個(gè)部分組成代碼段(.text)、數(shù)據(jù)段(.data)、BSS段(.bss),堆和棧組成,其中代碼段,數(shù)據(jù)段和BSS段是編譯的時(shí)候由編譯器分配的,而堆和 棧是程序運(yùn)行的時(shí)候由系統(tǒng)分配的。布局如下
在上圖中,由編譯器分配的地址空間都是在連接的時(shí)候分配的,而運(yùn)行時(shí)分配的空間是在程序運(yùn)行時(shí)由系統(tǒng)分配的
BSS段:BSS段(bss segment)通常是指用來(lái)存放程序中未初始化的全局變量和靜態(tài)變量 (這里注意一個(gè)問(wèn)題:一般的書(shū)上都會(huì)說(shuō)全局變量和靜態(tài)變量是會(huì)自動(dòng)初始化的,那么哪來(lái)的未初始化的變量呢?變量的初始化可以分為顯示初始化和隱式初始化,全局變量和靜態(tài)變量如果程序員自己不初始化的話的確也會(huì)被初始化,那就是不管什么類型都初始化為0,這種沒(méi)有顯示初始化的就是我們這里所說(shuō)的未初始化。既然都是0那么就沒(méi)必要把每個(gè)0都存儲(chǔ)起來(lái),從而節(jié)省磁盤(pán)空間,這是BSS的主要作用)的一塊內(nèi)存區(qū)域。BSS是英文Block Started by Symbol的簡(jiǎn)稱。BSS段屬于靜態(tài)內(nèi)存分配。 BSS節(jié)不包含任何數(shù)據(jù),只是簡(jiǎn)單的維護(hù)開(kāi)始和結(jié)束的地址,即總大小,以便內(nèi)存區(qū)能在運(yùn)行時(shí)分配并被有效地清零。BSS節(jié)在應(yīng)用程序的二進(jìn)制映象文件中并不存在,即不占用磁盤(pán)空間 而只在運(yùn)行的時(shí)候占用內(nèi)存空間 ,所以如果全局變量和靜態(tài)變量未初始化那么其可執(zhí)行文件要小很多。
數(shù)據(jù)段:數(shù)據(jù)段(data segment)通常是指用來(lái)存放程序中已初始化的全局變量和靜態(tài)變量的一塊內(nèi)存區(qū)域。數(shù)據(jù)段屬于靜態(tài)內(nèi)存分配,可以分為只讀數(shù)據(jù)段和讀寫(xiě)數(shù)據(jù)段。 字符串常量等,但一般都是放在只讀數(shù)據(jù)段中 。
代碼段:代碼段(code segment/text segment)通常是指用來(lái)存放程序執(zhí)行代碼的一塊內(nèi)存區(qū)域。這部分區(qū)域的大小在程序運(yùn)行前就已經(jīng)確定,并且內(nèi)存區(qū)域通常屬于只讀, 某些架構(gòu)也允許代碼段為可寫(xiě),即允許修改程序。在代碼段中,也有可能包含一些只讀的常數(shù)變量,例如字符串常量等,但一般都是放在只讀數(shù)據(jù)段中 。
堆(heap):堆是用于存放進(jìn)程運(yùn)行中被動(dòng)態(tài)分配的內(nèi)存段,它的大小并不固定,可動(dòng)態(tài)擴(kuò)張或縮減。當(dāng)進(jìn)程調(diào)用malloc等函數(shù)分配內(nèi)存時(shí),新分配的內(nèi)存就被動(dòng)態(tài)添加到堆上(堆被擴(kuò)張);當(dāng)利用free等函數(shù)釋放內(nèi)存時(shí),被釋放的內(nèi)存從堆中被剔除(堆被縮減)
棧 (stack):棧又稱堆棧, 是用戶存放程序臨時(shí)創(chuàng)建的局部變量,也就是說(shuō)我們函數(shù)括弧“{}”中定義的變量(但不包括static聲明的變量,static意味著在數(shù)據(jù)段中存放變 量)。除此以外,在函數(shù)被調(diào)用時(shí),其參數(shù)也會(huì)被壓入發(fā)起調(diào)用的進(jìn)程棧中,并且待到調(diào)用結(jié)束后,函數(shù)的返回值也會(huì)被存放回棧中。由于棧的先進(jìn)先出特點(diǎn),所以 棧特別方便用來(lái)保存/恢復(fù)調(diào)用現(xiàn)場(chǎng)。從這個(gè)意義上講,我們可以把堆??闯梢粋€(gè)寄存、交換臨時(shí)數(shù)據(jù)的內(nèi)存區(qū)。注意:??臻g是向下增長(zhǎng)的,每個(gè)線程有一個(gè)自己的棧,在linux上默認(rèn)的大小是8M,可以用ulimit查看和修改。
棧系統(tǒng)提供的功能,特點(diǎn)是快速高效,缺點(diǎn)是有限制,數(shù)據(jù)不靈活;而堆是函數(shù)庫(kù)提供的功能,特點(diǎn)是靈活方便,數(shù)據(jù)適應(yīng)面廣泛,但是效率有一定降低。
以下是一個(gè)簡(jiǎn)單的c文件,環(huán)境是OS--Linux,ARCH--PPC
##sta.c### #include <stdio.h> int kk[100] = {1,2,3,4,5}; int tt[100]; int ii; int main() { int i; static int si; char a[10]= "abcd"; printf("i is %d/n"); return 0; } 經(jīng)過(guò)gcc -S sta.c之后,生成的匯編代碼如下 ##sta.s### .file "sta.c" .gnu_attribute 4, 2 .gnu_attribute 8, 3 .globl kk .section ".data" .align 2 .type kk, @object .size kk, 400 kk: .long 1 .long 2 .long 3 .long 4 .long 5 .zero 380 .lcomm si.2254,4,4 .type si.2254, @object .section .rodata .align 2 .LC1: .string "i is %d/n" .align 2 .LC0: .string "abcd" .zero 5 .section ".text" .align 2 .globl main .type main, @function main: stwu 1,-32(1) mflr 0 stw 0,36(1) stw 31,28(1) mr 31,1 lis 9,.LC0@ha la 9,.LC0@l(9) lwz 0,0(9) lbz 9,4(9) stw 0,12(31) stb 9,16(31) li 0,0 stb 0,17(31) li 0,0 stb 0,18(31) li 0,0 stb 0,19(31) li 0,0 stb 0,20(31) li 0,0 stb 0,21(31) lis 9,.LC1@ha la 3,.LC1@l(9) crxor 6,6,6 bl printf li 0,0 mr 3,0 lwz 11,0(1) lwz 0,4(11) mtlr 0 lwz 31,-4(11) mr 1,11 blr .size main, .-main .comm tt,400,4 .comm ii,4,4 .ident "GCC: (GNU) 4.2.3" .section .note.GNU-stack,"",@progbits
Note: 一般編譯器和操作系統(tǒng)實(shí)現(xiàn)來(lái)說(shuō),對(duì)于虛擬地址空間的最低(從0開(kāi)始的幾K)的一段空間是未被映射的,也就是說(shuō)它在進(jìn)程空間中,但沒(méi)有賦予物理地址,不能被訪問(wèn)。這也就是對(duì)空指針的訪問(wèn)會(huì)導(dǎo)致crash的原因 ,因?yàn)榭罩羔樀牡刂肥?。至于為什么預(yù)留的不是一個(gè)字節(jié)而是幾K,是因?yàn)閮?nèi)存是分頁(yè)的,至少要一頁(yè);另外幾k的空間還可以用來(lái)捕捉使用空指針的情況。
上一篇:C++中replace()函數(shù)使用方法匯總
欄 目:C語(yǔ)言
本文標(biāo)題:通過(guò)一個(gè)小例子來(lái)簡(jiǎn)單理解C語(yǔ)言中的內(nèi)存空間管理
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/2657.html
您可能感興趣的文章
- 01-10c語(yǔ)言 跳臺(tái)階問(wèn)題的解決方法
- 01-10如何判斷一個(gè)數(shù)是否為2的冪次方?若是,并判斷出來(lái)是多少次方
- 01-10如何判斷一個(gè)數(shù)是否為4的冪次方?若是,并判斷出來(lái)是多少次方
- 01-10異步http listener 完全并發(fā)處理懲罰http懇求的小例子
- 01-10數(shù)組中求第K大數(shù)的實(shí)現(xiàn)方法
- 01-10如何判斷一個(gè)整數(shù)的二進(jìn)制中有多少個(gè)1
- 01-10探討:將兩個(gè)鏈表非降序合并為一個(gè)鏈表并依然有序的實(shí)現(xiàn)方法
- 01-10用C++實(shí)現(xiàn)一個(gè)鏈?zhǔn)綏5膶?shí)例代碼
- 01-10用C++實(shí)現(xiàn)strcpy(),返回一個(gè)char*類型的深入分析
- 01-10探討:用兩個(gè)棧實(shí)現(xiàn)一個(gè)隊(duì)列(我作為面試官的小結(jié))


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10C#中split用法實(shí)例總結(jié)