编译器对变量的内存分配方式

今天看到《C陷阱与缺陷》书中的一段代码:
int i,a[10];
for(i=1;i<=10;i++)
  a[i]=0;
书中说:在for语句的比较部分本来是i<10;却写成了i<=10;因此实际上并不存在的a[10]被设置为0,也就是内存在数组a之后的一个字(word)的内存被设置为0。如果用来编译这段程序的编译器按照内存地址递减的方式来给变量分配内存,那么内存中数组a之后的一个字(word)实际上是分配给了整型变量i。此时本来循环计数器i的值为10,循环体内将并不存在的a[10]设置为0,实际上却是将计数器i的值设置为0,这就陷入死循环。

于是为了验证编译器按照内存递减的方式给变量分配内存,写了一个测试例程
在VC6.0编译器下:
#include <iostream.h>
void main()
{
int i,j;
cout<<&i<<’ '<<&j;
}
结果输出为:0x12FF7C 0x12FF78,此编译器下对变量的分配方式安装地址递减的方式分配

在linux中的gcc编译器的测试程序1:
#include <stdio.h>
#include <stdlib.h>
int main()
{
int i,j;
printf(“i addr:%p j addr:%p\n”,&i,&j);
return 0;
}
输出结果如下:
结果输出为:0x12FF7C   0x12FF78
输出结果相反,编译器对变量的内存分配方式按照地址递增的方式分配。

在VC6.0编译器环境下的测试程序2:
#include <iostream.h>
void main()
{
int i,a[10];
cout<<&i<<endl<<&a[0]<<’ ‘<<&a[1]<<’ ‘<<&a[2]<<’ ‘<<&a[3]<<&a[4]<<endl <<&a[5]<<’ ‘<<&a[6]<<’ ‘<<&a[7]<<’ ‘<<&a[8]<<’ '<<&a[9]<<endl;
}
结果输出为:0x0012FF7C 0x0012FF54 0x0012FF58 0x0012FF5C 0x0012FF60 0x0012FF64 0x0012FF68 0x0012FF6C 0x0012FF70 0x0012FF74 0x0012FF78
可见对于语句int i,a[10];编译器在分配内存地址时也是按照内存地址递减的方式进行分配;因此在内存中实际的变量地址分配为从&a[0]~&a[9],紧接着是&i,即:
0x0012FF54 0x0012FF58 0x0012FF5C 0x0012FF60 0x0012FF64 0x0012FF68 0x0012FF6C 0x0012FF70 0x0012FF74 0x0012FF78 0x0012FF7C
此时就好理解原文中的后半句:因为实际上并不存在a[10],理论上的a[10]就是数组a之后的一个内存地址,由于编译器按照内存地址递减的方式给变量分配内存,这正好就是变量i的地址,即&a[10]==&i;所以循环体中a[10]=0;当i等于10时,a[10]=0;实则为i=0。为了验证所想是否正确,继续做了以下试验。

在Linux环境下gcc编译器下的测试程序2
#include <stdio.h>
#include <stdlib.h>
void main()
{
int i,a[10];
printf(“i addr:%p a[0]:%p a[1]:%p a[2]:%p a[3]:%p a[4]:%p a[5]:%p a[6]:%p a[7]:%p a[8]:%pa[9]:%pa[10]:%p\n”,&i,&a[0],&a[1],&a[2],&a[3],&a[4],&a[5],&a[6],&a[7],&a[8],&a[9],&a[10]);
}
在这里插入图片描述
从输出的变量地址结果值可以看出和在VC6.0环境下输出结果是相反的,在Linux环境下gcc编译器对变量的内存分配方式是按递增地址方式进行分配的。

在VC6.0的环境下测试程序3:
#include <iostream.h>
void main() {
int i,a[10];
cout<<&i<<’ '<<&a[10];
}
结果输出为:0x0012FF7C 0x0012FF7C
可见i与a[10]的内存地址确实是相同的。

在Linux环境下gcc编译的测试程序3
#include <stdio.h>
#include <stdlib.h>
void main()
{
int i,a[10];
printf(“i addr:%p a[10]:%p\n”,&i,&a[10]);
}
编译执行的结果如下
在这里插入图片描述
可见i和a[10]的地址并不一样。
因此在不同的编译器环境下,变量的内存分配方式是不一样的,可能最终的输出结果也会不一样,因此大家在使用的过程中药慎重考虑。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值