程序运行时堆区和栈区的分配
本文為轉(zhuǎn)載,由孫曉明老師所編寫,放在我的博客里面主要是為了自己學(xué)習(xí)方便。
棧(stack):是自動(dòng)分配變量,以及函數(shù)調(diào)用所使用的一些空間(所謂的局部變量),地址由高向低減少;?
堆(heap):由malloc,new等分配的空間的地址,地址由低向高增長(zhǎng)。?
看看下面一段代碼:?
#include <iostream>?
using namespace std;?
????????????void main()?
????????????{?
????????????int a;??
????????????int b;?
????????????int *p1?
????????????int *p2;??
????????????cout << &a << endl << &b << endl << &p1 << endl << &p2 << endl ;??
????????????//結(jié)果顯示:0012FF7C,0012FF78,0012FF74,0012FF70;可見,棧中的地址是減少的?
????????????int *p3=new int[1];??
????????????int *p4=new int[1];?
????????????cout<<p3<<endl<<p4<<endl;?
????????????//結(jié)果顯示:003907A8,003907E0;可見,堆中地址是增加的??
????????????}?
有一點(diǎn)應(yīng)該注意:頻繁使用heap 會(huì)產(chǎn)生內(nèi)存碎片,而按照堆棧的先入后出原則,即先申請(qǐng)的后釋放原則可以有效地避免在堆中產(chǎn)生碎片。?
棧區(qū)中的局部數(shù)據(jù)占有的空間在函數(shù)結(jié)束后會(huì)自動(dòng)釋放,而堆中的要由程序員手動(dòng)釋放。?
堆和棧中的存儲(chǔ)內(nèi)容??
Stack棧: 在函數(shù)調(diào)用時(shí),第一個(gè)進(jìn)棧的是主函數(shù)中? 函數(shù)調(diào)用語(yǔ)句的下一條可執(zhí)行語(yǔ)句的地址,然后是函數(shù)的各個(gè)參數(shù),然后是函數(shù)中的局部變量。注意靜態(tài)變量是不入棧的。??
當(dāng)本次函數(shù)調(diào)用結(jié)束后,局部變量先出棧,然后是參數(shù),最后棧頂指針指向最開始存的地址,也就是主函數(shù)中的下一條指令,程序由該點(diǎn)繼續(xù)運(yùn)行。???
heap堆:一般是在堆的頭部用一個(gè)字節(jié)存放堆的大小。堆中的具體內(nèi)容由程序員安排。?
程序的內(nèi)存分配??
一個(gè)由C/C++編譯的程序占用的內(nèi)存分為以下幾個(gè)部分??
1、棧區(qū)(stack)— 由編譯器自動(dòng)分配釋放 ,存放函數(shù)的參數(shù)值,局部變量的值等。與數(shù)據(jù)結(jié)構(gòu)棧類似?
2、堆區(qū)(heap) — 一般由程序員分配釋放, 若程序員不釋放,程序結(jié)束時(shí)可能由OS回收 。注意它與數(shù)據(jù)結(jié)構(gòu)中的堆是兩回事。?
3、全局區(qū)(靜態(tài)區(qū))(static)—全局變量和靜態(tài)變量的存儲(chǔ)是放在一塊的,初始化的全局變量和靜態(tài)變量在一塊區(qū)域, 未初始化的全局變量和未初始化的靜態(tài)變量在相鄰的另一塊區(qū)域。 - 程序結(jié)束后有系統(tǒng)釋放?
4、文字常量區(qū) —常量字符串就是放在這里的。 程序結(jié)束后由系統(tǒng)釋放??
5、程序代碼區(qū)(text)—存放函數(shù)體的二進(jìn)制代碼。?
引用的一個(gè)例子:?
????????????//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);??
????????????//分配得來(lái)得10和20字節(jié)的區(qū)域就在堆區(qū)。?
????????????strcpy(p1,?"123456");? ?
????????????//123456\0放在常量區(qū),編譯器可能會(huì)將它與p3所指向的"123456"優(yōu)化成一塊。?
????????????}
總結(jié)
以上是生活随笔為你收集整理的程序运行时堆区和栈区的分配的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 关于ARM的22个常用概念
- 下一篇: UNIX 高手的 20 个习惯