免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
123下一页
最近访问板块 发新帖
查看: 8047 | 回复: 27
打印 上一主题 下一主题

堆栈溢出的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2007-09-14 16:58 |只看该作者 |倒序浏览
#include <stdio.h>
int main()
{
    int x,y,z;
    int i;
    int a[16];
    for(i=0;i<=16;i++){
        a=0;
        printf("%d,",i);
    }
}
   
请高手说一下这个函数的栈空间分布

论坛徽章:
0
2 [报告]
发表于 2007-09-14 17:34 |只看该作者
你把 i 的地址与a[16]的地址打印出来就知道是怎么回事了.

论坛徽章:
59
2015年亚洲杯之约旦
日期:2015-01-27 21:27:392015年亚洲杯之日本
日期:2015-02-06 22:09:41拜羊年徽章
日期:2015-03-03 16:15:432015年辞旧岁徽章
日期:2015-03-03 16:54:152015年迎新春徽章
日期:2015-03-04 09:50:282015元宵节徽章
日期:2015-03-06 15:50:392015年亚洲杯之阿联酋
日期:2015-03-19 17:39:302015年亚洲杯之中国
日期:2015-03-23 18:52:23巳蛇
日期:2014-12-14 22:44:03双子座
日期:2014-12-10 21:39:16处女座
日期:2014-12-02 08:03:17天蝎座
日期:2014-07-21 19:08:47
3 [报告]
发表于 2007-09-14 19:35 |只看该作者
原帖由 soulmachine 于 2007-9-14 16:58 发表
#include
int main()
{
    int x,y,z;
    int i;
    int a[16];
    for(i=0;i


a=0;
出错:因为对于编译器来说,它是个恒量。
int x,y,z;  //32[ebp] base=20
int i;        //20[ebp]  base=16
int a[16] //16[ebp]   base=0
//如果这里还有变量
现在,你想让 0=0吗?
PS'0=0只是巧合,如果后面还有变量,比如
int b,c;
则变为   8[ebp],a的base=8
对于编译器说,也就相当于执行:
8=0;//这个语句能执行吗?

所以,该程序出错.

论坛徽章:
0
4 [报告]
发表于 2007-09-14 19:41 |只看该作者
原帖由 folklore 于 2007-9-14 19:35 发表


a=0;
出错:因为对于编译器来说,它是个恒量。
int x,y,z;  //32[ebp] base=20
int i;        //20[ebp]  base=16
int a[16] //16[ebp]   base=0
//如果这里还有变量
现在,你想让 0=0吗?
PS'0=0只 ...


楼主那个a=0 本来是没有写错, 应该是论坛的原因,其本来应该是写

  1. a[i] = 0;
复制代码


其实楼主想看是是不是这个东西, 即对于下面的代码:

  1. int i, a[10];
  2. for (i = 1; i <= 10; i++)
  3. {
  4.       a[i] = 0;
  5. }
复制代码

对于GCC编译器而言, 其为一个无限循环.
在这种情况下, a[10] 与 i 的地址相同了.

论坛徽章:
0
5 [报告]
发表于 2007-09-14 21:09 |只看该作者
楼上正解。
数组越界导致 i 被覆盖了。

论坛徽章:
0
6 [报告]
发表于 2007-09-14 22:12 |只看该作者
这个程序用GCC和VC编译后的结果是一样的。
    我打印了各个变量的地址,
printf("%d,%d,%d,%d,%d,%d\n",&x,&y,&z,&i,a,a+15);
结果为:
2280668,2280664,2280660,2280656,2280592,2280652
看来编译器分配变量的顺序与变量定义的顺序是相反的啊。它先为a[1]到a[15]分配空间,在接着为i,z,y,x分配空间。
     既然GCC和VC都是这样分配的,那么这种分配的顺序是与编译器自己实现的(恰好GCC和VC都用同一种方式而已),还是C语言标准里面规定的?

论坛徽章:
0
7 [报告]
发表于 2007-09-14 22:56 |只看该作者
楼主看来对栈还没学到家吧。
栈是由高地址向低地址增长的,所以还是先分配 i 等几个变量再分配 a 这个数组的。

论坛徽章:
0
8 [报告]
发表于 2007-09-14 23:47 |只看该作者
应该是先声明先分配吧,x,y,z先分配,i次之,然后数组a。只是由于分配时由高地址向底地址分配,所以从C代码来看,a[16]就是i的最后一个字节了,所以应该是个死循环。
这里编译器在实现的时候会有空挡吗?比如声明的不是a[16],是a[15]呢?猜测是没有,空到下面去了。

论坛徽章:
0
9 [报告]
发表于 2007-09-15 00:35 |只看该作者
我晕,我隐约记得上编译远离的时候,以及几个场合上见过说栈是从高地址向低地址增长的。但是大家看看APUE1的P127,里面有张图,说的是C程序的存储空间布局。由低地址到高地址分别是正文、初始化的数据、未初始化的数据(bss)、堆、栈。像这样的话,栈是怎么实现有高地址向低地址分配的呢?

论坛徽章:
0
10 [报告]
发表于 2007-09-15 10:21 |只看该作者
没错。
既然栈是处于最高的地址,那它怎么还能再向高地址增长呢?
至于怎么实现,那就得看操作系统实现和编译方面的书了。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP