程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> 關於C++ >> 淺析C/C++變量在內存中的散布

淺析C/C++變量在內存中的散布

編輯:關於C++

淺析C/C++變量在內存中的散布。本站提示廣大學習愛好者:(淺析C/C++變量在內存中的散布)文章只能為提供參考,不一定能成為您想要的結果。以下是淺析C/C++變量在內存中的散布正文


C/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("堆中數據地址:0x%08x\n", p); 

    putchar('\n'); 
    printf("棧中數據地址(有初值):0x%08x = %d\n", &i, i); 
    printf("棧中數據地址(有初值):0x%08x = %d\n", &j, j); 
    printf("棧中數據地址(有初值):0x%08x = %d\n", &k, k); 
    printf("棧中數據地址(無初值):0x%08x = %d\n", &f, f); 
    printf("棧中數據地址(無初值):0x%08x = %d\n", &h, h); 

    putchar('\n'); 
    printf("靜態數據地址(有初值):0x%08x = %d\n", &s_i, s_i); 
    printf("靜態數據地址(有初值):0x%08x = %d\n", &s_j, s_j); 
    printf("靜態數據地址(無初值):0x%08x = %d\n", &s_k, s_k); 
    printf("靜態數據地址(無初值):0x%08x = %d\n", &s_h, s_h); 

    putchar('\n'); 
    printf("全局數據地址(有初值):0x%08x = %d\n", &g_i, g_i); 
    printf("全局數據地址(有初值):0x%08x = %d\n", &g_j, g_j); 
    printf("全局數據地址(無初值):0x%08x = %d\n", &g_k, g_k); 
    printf("全局數據地址(無初值):0x%08x = %d\n", &g_h, g_h); 

    putchar('\n'); 
    printf("字符串常量數據地址:0x%08x 指向 0x%08x 內容為-%s\n", &pstr1, pstr1, pstr1); 
    printf("字符串常量數據地址:0x%08x 指向 0x%08x 內容為-%s\n", &pstr2, pstr2, pstr2); 
    printf("字符串常量數據地址:0x%08x 指向 0x%08x 內容為-%s\n", &pstr3, pstr3, pstr3); 
    free(p); 
    return 0; 


運轉成果(Release版本,XP體系)以下:

可以看出:
1. 變量在內存地址的散布為:堆-棧-代碼區-全局靜態-常量數據
2. 統一區域的各變量按聲明的次序在內存的中順次由低到高分派空間(只要未賦值的全局變量是個破例)
3. 全局變量和靜態變量假如不賦值,默許為0。 棧中的變量假如不賦值,則是一個隨機的數據。
4. 編譯器會以為全局變量和靜態變量是同等的,已初始化的全局變量和靜態變量分派在一路,未初始化的全局變量和靜態變量分派在另外一起。

下面法式全在一個主函數中,上面增長函數挪用,看看函數的參數和函數中變量會分派在甚麼處所。

法式以下:

#include <stdio.h> 
void fun(int i) 

    int j = i; 
    static int s_i = 100; 
    static int s_j; 

    printf("子函數的參數:        0x%p = %d\n", &i, i); 
    printf("子函數 棧中數據地址: 0x%p = %d\n", &j, j); 
    printf("子函數 靜態數據地址(有初值): 0x%p = %d\n", &s_i, s_i); 
    printf("子函數 靜態數據地址(無初值): 0x%p = %d\n", &s_j, s_j); 

int main() 

    int i = 5; 
    static int s_i = 100; 
    static int s_j; 

    printf("主函數 棧中數據地址: 0x%p = %d\n", &i, i); 
    printf("主函數 靜態數據地址(有初值): 0x%p = %d\n", &s_i, s_i); 
    printf("子函數 靜態數據地址(無初值): 0x%p = %d\n", &s_j, s_j); 
    putchar('\n'); 

    fun(i); 
    return 0; 


運轉成果以下:

可以看出,主函數中棧的地址都要高於子函數中參數及棧地址,證實了棧的舒展偏向是由窪地址向低地址擴大的。主函數和子函數中靜態數據的地址也是相鄰的,解釋法式會將已初始化的全局變量和靜態變量分派在一路,未初始化的全局變量和靜態變量分派在一路。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved