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 = "u014053368123456789";
	char *pstr2 = "u014053368123456789";
	char *pstr3 = "Hello";

	printf("堆中数据地址:0x%08x\n", p);
	free(p);
	p=(int*)malloc(MAXN * sizeof(int));
	printf("堆中数据地址:0x%08x\n", p);
	int* p2=(int*)malloc(MAXN * sizeof(int));
	printf("堆中数据地址:0x%08x\n", p2);
	free(p);
	free(p2);

	putchar('\n');
	printf("栈中常量数据的地址:0x%08x = %d\n", &MAXN, MAXN);

	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);
	
	return 0;
}

 

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


可以看出:

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

2.      同一区域的各变量按声明的顺序在内存的中依次由高到低分配空间(已初始化的静态数据和已初始化的全局数据是个例外)。

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

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

5.      在堆上,各变量按顺序在内存中依次由低到高分配空间。而且堆内存使用完后,释放堆内存,下次再次分配堆内存来存储数据,会发现地址还是一样的。

 

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

程序如下:

#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
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值