浅析C/C++变量在内存中的分布

 更新时间:2013年09月14日 09:44:31   作者:  
变量在内存地址的分布为:堆-栈-代码区-全局静态-常量数据。同一区域的各变量按声明的顺序在内存的中依次由低到高分配空间(只有未赋值的全局变量是个例外)

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; 


运行结果如下:

可以看出,主函数中栈的地址都要高于子函数中参数及栈地址,证明了栈的伸展方向是由高地址向低地址扩展的。主函数和子函数中静态数据的地址也是相邻的,说明程序会将已初始化的全局变量和静态变量分配在一起,未初始化的全局变量和静态变量分配在一起。

相关文章

  • 如何在C++中实现按位存取

    如何在C++中实现按位存取

    实现紧凑存取,不是按一个字节一个字节地存取,而是按位存取,本文就是介绍了如何在C++中实现按位存取,需要的朋友可以参考下
    2015-07-07
  • 浅谈C++重载、重写、重定义

    浅谈C++重载、重写、重定义

    下面小编就为大家带来一篇浅谈C++重载、重写、重定义。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-06-06
  • C ++迭代器iterator在string中使用方法介绍

    C ++迭代器iterator在string中使用方法介绍

    迭代器是一种检查容器内元素并遍历元素的数据类型。迭代器是一个变量,提供对一个容器中的对象的(间接)访问方法,并且定义了容器中对象的范围。迭代器可以指向容器中的某个元素,通过迭代器就可以对非数组(存储空间不连续)的数据结构进行遍历
    2022-10-10
  • Clion2020.2.x最新激活码破解版附安装教程(Mac Linux Windows)

    Clion2020.2.x最新激活码破解版附安装教程(Mac Linux Windows)

    Clion2020增加了很多新特性,修复了大量bug,大大提高了开发效率。这篇文章主要介绍了Clion2020.2.x最新激活码破解版附安装教程(Mac Linux Windows),需要的朋友可以参考下
    2020-11-11
  • 深入解析C++设计模式编程中解释器模式的运用

    深入解析C++设计模式编程中解释器模式的运用

    这篇文章主要介绍了C++设计模式编程中解释器模式的运用,解释器模式给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子,需要的朋友可以参考下
    2016-03-03
  • 解析ActiveMQ的使用说明总结

    解析ActiveMQ的使用说明总结

    本篇文章是对ActiveMQ的使用进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C语言全面讲解顺序表使用操作

    C语言全面讲解顺序表使用操作

    线性表是最简单的数据结构,而顺序表又是最简单的线性表,其基本思想是用一段地址连续的储存单元依次存储线性表的数据元素,比如我们常用的一维数组,下面代码实现了顺序表的定义以及基本操作
    2022-04-04
  • C语言中 printf 函数输出格式

    C语言中 printf 函数输出格式

    这篇文章主要介绍了C语言中 printf 函数简介,通过实例代码给大家介绍Printf输出格式的相关知识,需要的朋友可以参考下
    2021-08-08
  • C语言实现外卖管理系统

    C语言实现外卖管理系统

    这篇文章主要为大家详细介绍了C语言实现外卖管理系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-11-11
  • C++11智能指针unique_ptr用法使用场景分析

    C++11智能指针unique_ptr用法使用场景分析

    unique_ptr 是 C++ 11 提供的用于防止内存泄漏的智能指针中的一种实现,即使在异常发生时也可帮助避免资源泄露。这篇文章主要介绍了C++11智能指针unique_ptr用法介绍,需要的朋友可以参考下
    2021-08-08

最新评论