浅析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; 
}

运行结果如下:

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

(0)

相关推荐

  • 深入理解c/c++ 内存对齐

    内存对齐,memory alignment.为了提高程序的性能,数据结构(尤其是栈)应该尽可能地在自然边界上对齐.原因在于,为了访问未对齐的内存,处理器需要作两次内存访问:然而,对齐的内存访问仅需要一次访问.内存对齐一般讲就是cpu access memory的效率(提高运行速度)和准确性(在一些条件下,如果没有对齐会导致数据不同步现象).依赖cpu,平台和编译器的不同.一些cpu要求较高(这句话说的不准确,但是确实依赖cpu的不同),而有些平台已经优化内存对齐问题,不同编译器的对齐模数不同.总

  • 深入解析C++ Data Member内存布局

    如果一个类只定义了类名,没定义任何方法和字段,如class A{};那么class A的每个实例占用1个字节的内存,编译器会会在这个其实例中安插一个char,以保证每个A实例在内存中有唯一的地址,如A a,b;&a!=&b.如果一个直接或是间接的继承(不是虚继承)了多个类,如果这个类及其父类像A一样没有方法没有字段,那么这个类的每个实例的大小都是1字节,如果有虚继承,那就不是1字节了,每虚继承一个类,这个类的实例就会多一个指向被虚继承父类的指针.还有一点值得说明的就是像A这样的类,编译器不

  • 基于C++中常见内存错误的总结

    在系统开发过程中出现的bug相对而言是比较好解决的,花费在这个上面的调试代价不是很大,但是在系统集成后的bug往往是难以定位的bug(最好方式是打桩,通过打桩可以初步锁定出错的位置,如:进入函数前打印日志,离开时再次打印日志).而这些难以定位的bug基本分为2类:内存错误和并非问题. 1.内存泄露如果在堆栈上分配的内存使用完成后没有释放就会造成内存泄露.少量的内存泄露不至于让程序崩溃,但是大量的内存泄露就会导致内存耗尽,后续内存分配失败,从而导致程序崩溃.长时间运行软件,即使只有一两处泄露,同样

  • C/C++动态分配与释放内存的区别详细解析

    1. malloc()函数1.1 malloc的全称是memory allocation,中文叫动态内存分配.原型:extern void *malloc(unsigned int num_bytes); 说明:分配长度为num_bytes字节的内存块.如果分配成功则返回指向被分配内存的指针,分配失败返回空指针NULL.当内存不再使用时,应使用free()函数将内存块释放. 1.2 void *malloc(int size); 说明:malloc 向系统申请分配指定size个字节的内存空间,返

  • C/C++语言中结构体的内存分配小例子

    当未用 #pragma 指令指定编译器的对齐位数时,结构体按最长宽度的数据成员的宽度对齐:当使用了 #pragma 指令指定编译器的对齐位数时,结构体按最长宽度的数据成员的宽度和 #pragma 指令指定的位数中的较小值对齐. #pragma 指令格式如下所示:#pragma pack(4)     // 或者 #pragma pack(push, 4) 举例如下:(机器字长为 32 位)    struct    {        char a;    }test;    printf("%d

  • c++实现逐行读取配置文件写入内存的示例

    不解析配置内容,只读取文件内容,剪去注释和首尾空格后写入缓存: vector<string> 中.供其他方法使用.代码是在做一个MFC小工具时写的. ReadProtocol.h 复制代码 代码如下: /*** 从文件中 读取 protocol 的内容 写入缓存* 供外部方法使用* Alex Liu, 2014*/ #pragma once #include <vector>#include <map>#include <list>#include <

  • C++中给二维指针分配内存(实现代码)

    原理就不写在这里了,毕竟网上的介绍有很多,代码如下所示: 复制代码 代码如下: #include <iostream>using namespace std; #define  N  5#define  M  10 int main(int argc, char **argv){ int **p; int i,j; p = new int* [N]; for (i = 0; i < N; i++)  p[i] = new int [M]; for (i = 0; i < N; i+

  • C++内存查找实例

    本文实例讲述了C++内存查找的方法,分享给大家供大家参考.具体如下: windows程序设计中的内存查找功能,主程序代码如下: 复制代码 代码如下: // MemRepair.cpp : 定义控制台应用程序的入口点.  //    #include "stdafx.h"  #include <Windows.h>    BOOL FindFirst(DWORD dwValue);  BOOL FindNext(DWORD dwValue);  HANDLE g_hProce

  • VC++中内存对齐实例教程

    内存对其是VC++程序设计中一个非常重要的技巧,本文即以实例讲述VC++实现内存对其的方法.具体分析如下: 一.概述 我们经常看到求 sizeof(A) 的值的问题,其中A是一个结构体,类,或者联合体. 为了优化CPU访问和优化内存,减少内存碎片,编译器对内存对齐制定了一些规则.但是,不同的编译器可能有不同的实现,本文只针对VC++编译器,这里使用的IDE是VS2012. #pragma pack()是一个预处理,表示内存对齐.布局控制#pragma,为编译程序提供非常规的控制流信息. 二.结构

  • c++动态内存空间示例(自定义空间类型大小和空间长度)

    动态内存空间的申请示范 利用C++的特性,能够自定义空间的类型大小和空间长度 下面这个程序是个数组动态配置的简单示例 复制代码 代码如下: #include <iostream>using namespace std; int main(){   int size = 0; cout << "请输入数组长度:";  //能够自定义的动态申请空间长度    cin >> size;    int *arr_Point = new int[size];

  • C语言、C++内存对齐问题详解

    这也可以? 复制代码 代码如下: #include <iostream> using namespace std;   struct Test_A {      char a;      char b;      int c; };   struct Test_B {      char a;      int c;      char b; };   struct Test_C {      int c;      char a;      char b; };   int main() {

  • C/C++ 传递动态内存的深入理解

    当你涉及到C/C++的核心编程的时候,你会无止境地与内存管理打交道.这些往往会使人受尽折磨.所以如果你想深入C/C++编程,你必须静下心来,好好苦一番.现在我们将讨论C/C++里我认为哪一本书都没有完全说清楚,也是涉及概念细节最多,语言中最难的技术之一的动态内存的传递.并且在软件开发中很多专业人员并不能写出相关的合格的代码.[引入] 看下面的例子,这是我们在编写库函数或者项目内的共同函数经常希望的. 复制代码 代码如下: void MyFunc(char *pReturn, size_t siz

随机推荐