C++堆和棧的區(qū)別與聯(lián)系講解
C++中,內(nèi)存分為5個(gè)區(qū):堆、棧、自由存儲區(qū)、全局/靜態(tài)存儲區(qū)和常量存儲區(qū)。
- 棧:是由編譯器在需要時(shí)自動分配,不需要時(shí)自動清除的變量存儲區(qū)。通常存放局部變量、函數(shù)參數(shù)等。
- 堆:是由new分配的內(nèi)存塊,由程序員釋放(編譯器不管),一般一個(gè)new與一個(gè)delete對應(yīng),一個(gè)new[]與一個(gè)delete[]對應(yīng)。如果程序員沒有釋放掉, 資源將由操作系統(tǒng)在程序結(jié)束后自動回收。
- 自由存儲區(qū):是由malloc等分配的內(nèi)存塊,和堆十分相似,用free來釋放。
- 全局/靜態(tài)存儲區(qū):全局變量和靜態(tài)變量被分配到同一塊內(nèi)存中(在C語言中,全局變量又分為初始化的和未初始化的,C++中沒有這一區(qū)分)。
- 常量存儲區(qū):這是一塊特殊存儲區(qū),里邊存放常量,不允許修改。
(注意:堆和自由存儲區(qū)其實(shí)不過是同一塊區(qū)域,new底層實(shí)現(xiàn)代碼中調(diào)用了malloc,new可以看成是malloc智能化的高級版本)
一. 堆與棧的討論:
- 管理方式:堆中資源由程序員控制(容易產(chǎn)生memory leak), 棧資源由編譯器自動管理,無需手工控制。
- 系統(tǒng)響應(yīng):對于堆,應(yīng)知道系統(tǒng)有一個(gè)記錄空閑內(nèi)存地址的鏈表,當(dāng)系統(tǒng)收到程序申請時(shí),遍歷該鏈表,尋找第一個(gè)空間大于申請空間的堆結(jié)點(diǎn),刪 除空閑結(jié)點(diǎn)鏈表中的該結(jié)點(diǎn),并將該結(jié)點(diǎn)空間分配給程序(大多數(shù)系統(tǒng)會在這塊內(nèi)存空間首地址記錄本次分配的大小,這樣delete才能正確釋放本內(nèi)存 空間,另外系統(tǒng)會將多余的部分重新放入空閑鏈表中)。對于棧,只要棧的剩余空間大于所申請空間,系統(tǒng)為程序提供內(nèi)存,否則報(bào)異常提示棧出。
- 空間大?。?nbsp;堆是不連續(xù)的內(nèi)存區(qū)域(因?yàn)橄到y(tǒng)是用鏈表來存儲空閑內(nèi)存地址,自然不是連續(xù)的),堆大小受限于計(jì)算機(jī)系統(tǒng)中有效的虛擬內(nèi)存(32bit 系統(tǒng)理論上是4G),所以堆的空間比較靈活,比較大。棧是一塊連續(xù)的內(nèi)存區(qū)域,大小是操作系統(tǒng)預(yù)定好的,windows下棧大小是2M(也有是1M,在 編譯時(shí)確定,VC中可設(shè)置)。
- 碎片問題:對于堆,頻繁的new/delete會造成大量碎片,使程序效率降低。 對于棧,它是一個(gè)先進(jìn)后出的隊(duì)列,進(jìn)出一一對應(yīng),不會產(chǎn)生碎片。
- 生長方向:堆向上,向高地址方向增長。棧向下,向低地址方向增長。
- 分配方式: 堆都是動態(tài)分配(沒有靜態(tài)分配的堆)。棧有靜態(tài)分配和動態(tài)分配,靜態(tài)分配由編譯器完成(如局部變量分配),動態(tài)分配由alloca函數(shù) 分 配,但棧的動態(tài)分配的資源由編譯器進(jìn)行釋放,無需程序員實(shí)現(xiàn)。
- 分配效率:堆由C/C++函數(shù)庫提供,機(jī)制很復(fù)雜。所以堆的效率比棧低很多。棧是極其系統(tǒng)提供的數(shù)據(jù)結(jié)構(gòu),計(jì)算機(jī)在底層對棧提供支持,分配專門 寄存 器存放棧地址,棧操作有專門指令。
二. 程序示例
通過下面的程序可以更好的對上面的概念進(jìn)行理解。
int b;
//main.cpp
int a = 0; //全局初始化區(qū)
char *p1; //全局未初始化區(qū)
main(){int b; //棧
char s[] = "abc"; // 棧
char *p2; //棧
char *p3 = "123456"; // 123456/0在常量區(qū),p3在棧上。
static int c = 0; // 全局(靜態(tài))初始化區(qū)
p1 = (char *)malloc(10)
p2 = (char *)malloc(20) // 分配得來得10和20字節(jié)的區(qū)域就在堆區(qū)。
strcpy(p1, "123456"); // 123456/0放在常量區(qū),編譯器可能會將它與p3所指向的"123456"優(yōu)化成一個(gè)地方。
}
小結(jié):
堆和棧的區(qū)別可以用如下的比喻來看出:
使用棧就象我們?nèi)ワ堭^里吃飯,只管點(diǎn)菜(發(fā)出申請)、付錢、和吃(使用),吃飽了就走,不必理會切菜、洗菜等準(zhǔn)備工作和洗碗、刷鍋等掃尾工作,他的好處是快捷,但是自由度小。使用堆就象是自己動手做喜歡吃的菜肴,比較麻煩,但是比較符合自己的口味,而且自由度大。(經(jīng)典!)
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接
相關(guān)文章
詳解如何使用VSCode和CMake構(gòu)建跨平臺的C/C++開發(fā)環(huán)境
本文主要介紹了如何使用VSCode和CMake構(gòu)建跨平臺的C/C++開發(fā)環(huán)境,想進(jìn)行跨平臺開發(fā)的同學(xué)們,一定要看一下2021-06-06
C++實(shí)現(xiàn)LeetCode(103.二叉樹的之字形層序遍歷)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(103.二叉樹的之字形層序遍歷),本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07
C 程序?qū)崿F(xiàn)密碼隱秘輸入的實(shí)例 linux系統(tǒng)可執(zhí)行
下面小編就為大家?guī)硪黄狢 程序?qū)崿F(xiàn)密碼隱秘輸入的實(shí)例 linux系統(tǒng)可執(zhí)行。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-11-11
c語言統(tǒng)計(jì)素?cái)?shù)之和的實(shí)例
這篇文章主要介紹了c語言統(tǒng)計(jì)素?cái)?shù)之和的實(shí)例,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12
C++?ffmpeg實(shí)現(xiàn)將視頻幀轉(zhuǎn)換成jpg或png等圖片
有時(shí)播放實(shí)時(shí)流的時(shí)候有截圖的需求,需要將解碼出來的圖片保存本地或上傳服務(wù)器,這時(shí)就需要將avframe中的數(shù)據(jù)編碼成png、jpg等格式的圖片,我們使用ffmpeg的相關(guān)編碼器就可以實(shí)現(xiàn)功能,下面就來講講具體實(shí)現(xiàn)方法吧2023-03-03
C++棧(stack)的模板類實(shí)現(xiàn)代碼
這篇文章主要為大家詳細(xì)介紹了C++棧(stack)的模板類實(shí)現(xiàn)代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
C++超詳細(xì)分析講解內(nèi)聯(lián)函數(shù)
為了消除函數(shù)調(diào)用的時(shí)空開銷,C++ 提供一種提高效率的方法,即在編譯時(shí)將函數(shù)調(diào)用處用函數(shù)體替換,類似于C語言中的宏展開。這種在函數(shù)調(diào)用處直接嵌入函數(shù)體的函數(shù)稱為內(nèi)聯(lián)函數(shù)(Inline Function),又稱內(nèi)嵌函數(shù)或者內(nèi)置函數(shù)2022-06-06

