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

本文通过测试程序深入解析C/C++变量在内存中的分布情况,包括堆、栈、全局静态、常量数据区域的位置及其特性,以及函数调用时参数和变量的分配位置。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

原文地址:http://blog.youkuaiyun.com/morewindows/article/details/6851681

C/C++变量在内存中的分布在笔试时经常考到,虽然简单,但也容易忘记,因此在这作个总结,以加深印象。

先写一个测试程序:

  1. #include<stdio.h>
  2. #include<malloc.h>
  3. intg_i=100;
  4. intg_j=200;
  5. intg_k,g_h;
  6. intmain()
  7. {
  8. constintMAXN=100;
  9. int*p=(int*)malloc(MAXN*sizeof(int));
  10. staticints_i=5;
  11. staticints_j=10;
  12. staticints_k;
  13. staticints_h;
  14. inti=5;
  15. intj=10;
  16. intk=20;
  17. intf,h;
  18. char*pstr1="MoreWindows123456789";
  19. char*pstr2="MoreWindows123456789";
  20. char*pstr3="Hello";
  21. printf("堆中数据地址:0x%08x\n",p);
  22. putchar('\n');
  23. printf("栈中数据地址(有初值):0x%08x=%d\n",&i,i);
  24. printf("栈中数据地址(有初值):0x%08x=%d\n",&j,j);
  25. printf("栈中数据地址(有初值):0x%08x=%d\n",&k,k);
  26. printf("栈中数据地址(无初值):0x%08x=%d\n",&f,f);
  27. printf("栈中数据地址(无初值):0x%08x=%d\n",&h,h);
  28. putchar('\n');
  29. printf("静态数据地址(有初值):0x%08x=%d\n",&s_i,s_i);
  30. printf("静态数据地址(有初值):0x%08x=%d\n",&s_j,s_j);
  31. printf("静态数据地址(无初值):0x%08x=%d\n",&s_k,s_k);
  32. printf("静态数据地址(无初值):0x%08x=%d\n",&s_h,s_h);
  33. putchar('\n');
  34. printf("全局数据地址(有初值):0x%08x=%d\n",&g_i,g_i);
  35. printf("全局数据地址(有初值):0x%08x=%d\n",&g_j,g_j);
  36. printf("全局数据地址(无初值):0x%08x=%d\n",&g_k,g_k);
  37. printf("全局数据地址(无初值):0x%08x=%d\n",&g_h,g_h);
  38. putchar('\n');
  39. printf("字符串常量数据地址:0x%08x指向0x%08x内容为-%s\n",&pstr1,pstr1,pstr1);
  40. printf("字符串常量数据地址:0x%08x指向0x%08x内容为-%s\n",&pstr2,pstr2,pstr2);
  41. printf("字符串常量数据地址:0x%08x指向0x%08x内容为-%s\n",&pstr3,pstr3,pstr3);
  42. free(p);
  43. return0;
  44. }

运行结果(Release版本,XP系统)如下:

可以看出:

1. 变量在内存地址的分布为:堆-栈-代码区-全局静态-常量数据

2. 同一区域的各变量按声明的顺序在内存的中依次由低到高分配空间(只有未赋值的全局变量是个例外)。

3. 全局变量和静态变量如果不赋值,默认为0。栈中的变量如果不赋值,则是一个随机的数据。

4. 编译器会认为全局变量和静态变量是等同的,已初始化的全局变量和静态变量分配在一起,未初始化的全局变量和静态变量分配在另一起。

上面程序全在一个主函数中,下面增加函数调用,看看函数的参数和函数中变量会分配在什么地方。

程序如下:

  1. #include<stdio.h>
  2. voidfun(inti)
  3. {
  4. intj=i;
  5. staticints_i=100;
  6. staticints_j;
  7. printf("子函数的参数:0x%p=%d\n",&i,i);
  8. printf("子函数栈中数据地址:0x%p=%d\n",&j,j);
  9. printf("子函数静态数据地址(有初值):0x%p=%d\n",&s_i,s_i);
  10. printf("子函数静态数据地址(无初值):0x%p=%d\n",&s_j,s_j);
  11. }
  12. intmain()
  13. {
  14. inti=5;
  15. staticints_i=100;
  16. staticints_j;
  17. printf("主函数栈中数据地址:0x%p=%d\n",&i,i);
  18. printf("主函数静态数据地址(有初值):0x%p=%d\n",&s_i,s_i);
  19. printf("子函数静态数据地址(无初值):0x%p=%d\n",&s_j,s_j);
  20. putchar('\n');
  21. fun(i);
  22. return0;
  23. }

运行结果如下:

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


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值