淺析C/C++變量在內(nèi)存中的分布
C/C++變量在內(nèi)存中的分布在筆試時(shí)經(jīng)常考到,雖然簡(jiǎn)單,但也容易忘記,因此在這作個(gè)總結(jié),以加深印象。
先寫一個(gè)測(cè)試程序:
#include <stdio.h>
#include <malloc.h>
int g_i = 100;
int g_j = 200;
int g_k, g_h;
int main()
{
const int MAXN = 100;
int *p = (int*)malloc(MAXN * sizeof(int));
static int s_i = 5;
static int s_j = 10;
static int s_k;
static int s_h;
int i = 5;
int j = 10;
int k = 20;
int f, h;
char *pstr1 = "MoreWindows123456789";
char *pstr2 = "MoreWindows123456789";
char *pstr3 = "Hello";
printf("堆中數(shù)據(jù)地址:0x%08x\n", p);
putchar('\n');
printf("棧中數(shù)據(jù)地址(有初值):0x%08x = %d\n", &i, i);
printf("棧中數(shù)據(jù)地址(有初值):0x%08x = %d\n", &j, j);
printf("棧中數(shù)據(jù)地址(有初值):0x%08x = %d\n", &k, k);
printf("棧中數(shù)據(jù)地址(無(wú)初值):0x%08x = %d\n", &f, f);
printf("棧中數(shù)據(jù)地址(無(wú)初值):0x%08x = %d\n", &h, h);
putchar('\n');
printf("靜態(tài)數(shù)據(jù)地址(有初值):0x%08x = %d\n", &s_i, s_i);
printf("靜態(tài)數(shù)據(jù)地址(有初值):0x%08x = %d\n", &s_j, s_j);
printf("靜態(tài)數(shù)據(jù)地址(無(wú)初值):0x%08x = %d\n", &s_k, s_k);
printf("靜態(tài)數(shù)據(jù)地址(無(wú)初值):0x%08x = %d\n", &s_h, s_h);
putchar('\n');
printf("全局?jǐn)?shù)據(jù)地址(有初值):0x%08x = %d\n", &g_i, g_i);
printf("全局?jǐn)?shù)據(jù)地址(有初值):0x%08x = %d\n", &g_j, g_j);
printf("全局?jǐn)?shù)據(jù)地址(無(wú)初值):0x%08x = %d\n", &g_k, g_k);
printf("全局?jǐn)?shù)據(jù)地址(無(wú)初值):0x%08x = %d\n", &g_h, g_h);
putchar('\n');
printf("字符串常量數(shù)據(jù)地址:0x%08x 指向 0x%08x 內(nèi)容為-%s\n", &pstr1, pstr1, pstr1);
printf("字符串常量數(shù)據(jù)地址:0x%08x 指向 0x%08x 內(nèi)容為-%s\n", &pstr2, pstr2, pstr2);
printf("字符串常量數(shù)據(jù)地址:0x%08x 指向 0x%08x 內(nèi)容為-%s\n", &pstr3, pstr3, pstr3);
free(p);
return 0;
}
運(yùn)行結(jié)果(Release版本,XP系統(tǒng))如下:
可以看出:
1. 變量在內(nèi)存地址的分布為:堆-棧-代碼區(qū)-全局靜態(tài)-常量數(shù)據(jù)
2. 同一區(qū)域的各變量按聲明的順序在內(nèi)存的中依次由低到高分配空間(只有未賦值的全局變量是個(gè)例外)
3. 全局變量和靜態(tài)變量如果不賦值,默認(rèn)為0。 棧中的變量如果不賦值,則是一個(gè)隨機(jī)的數(shù)據(jù)。
4. 編譯器會(huì)認(rèn)為全局變量和靜態(tài)變量是等同的,已初始化的全局變量和靜態(tài)變量分配在一起,未初始化的全局變量和靜態(tài)變量分配在另一起。
上面程序全在一個(gè)主函數(shù)中,下面增加函數(shù)調(diào)用,看看函數(shù)的參數(shù)和函數(shù)中變量會(huì)分配在什么地方。
程序如下:
#include <stdio.h>
void fun(int i)
{
int j = i;
static int s_i = 100;
static int s_j;
printf("子函數(shù)的參數(shù): 0x%p = %d\n", &i, i);
printf("子函數(shù) 棧中數(shù)據(jù)地址: 0x%p = %d\n", &j, j);
printf("子函數(shù) 靜態(tài)數(shù)據(jù)地址(有初值): 0x%p = %d\n", &s_i, s_i);
printf("子函數(shù) 靜態(tài)數(shù)據(jù)地址(無(wú)初值): 0x%p = %d\n", &s_j, s_j);
}
int main()
{
int i = 5;
static int s_i = 100;
static int s_j;
printf("主函數(shù) 棧中數(shù)據(jù)地址: 0x%p = %d\n", &i, i);
printf("主函數(shù) 靜態(tài)數(shù)據(jù)地址(有初值): 0x%p = %d\n", &s_i, s_i);
printf("子函數(shù) 靜態(tài)數(shù)據(jù)地址(無(wú)初值): 0x%p = %d\n", &s_j, s_j);
putchar('\n');
fun(i);
return 0;
}
運(yùn)行結(jié)果如下:
可以看出,主函數(shù)中棧的地址都要高于子函數(shù)中參數(shù)及棧地址,證明了棧的伸展方向是由高地址向低地址擴(kuò)展的。主函數(shù)和子函數(shù)中靜態(tài)數(shù)據(jù)的地址也是相鄰的,說(shuō)明程序會(huì)將已初始化的全局變量和靜態(tài)變量分配在一起,未初始化的全局變量和靜態(tài)變量分配在一起。
相關(guān)文章
C語(yǔ)言實(shí)現(xiàn)紅黑樹的實(shí)例代碼
這篇文章主要介紹了C語(yǔ)言實(shí)現(xiàn)紅黑樹的實(shí)例代碼,有需要的朋友可以參考一下2013-12-12C++11的future和promise、parkged_task使用
這篇文章主要介紹了C++11的future和promise、parkged_task使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04C語(yǔ)言數(shù)據(jù)在內(nèi)存中的存儲(chǔ)詳解
本篇文章是C語(yǔ)言編程篇,主要為大家介紹C語(yǔ)言編程中數(shù)據(jù)在內(nèi)存中存儲(chǔ)解析,有需要的朋友可以借鑒參考下,希望可以有所幫助2021-09-09C語(yǔ)言開發(fā)實(shí)現(xiàn)貪吃蛇小游戲
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言開發(fā)實(shí)現(xiàn)貪吃蛇小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-10-10C++學(xué)習(xí)之虛函數(shù)表與多態(tài)詳解
這篇文章主要為大家詳細(xì)介紹了C++中虛函數(shù)表與多態(tài)的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C++有一定的幫助,感興趣的小伙伴可以了解一下2023-03-03Windows網(wǎng)絡(luò)編程之winsock實(shí)現(xiàn)文件傳輸示例
這篇文章主要介紹了Windows網(wǎng)絡(luò)編程之winsock實(shí)現(xiàn)文件傳輸示例,對(duì)于學(xué)習(xí)Windows網(wǎng)絡(luò)程序設(shè)計(jì)來(lái)說(shuō)具有很好的學(xué)習(xí)借鑒價(jià)值,需要的朋友可以參考下2014-08-08