转:http://blog.csdn.net/yucan1001/article/details/7450388
思科的一道面试题如下:
不要用编译器 用1分钟事件思考下
源文件A:int a[10];
int *b=a;
源文件B:
extern int *a; //表示其他文中已经定义了一个指针
extern int b[];//表示其他文中已经定义了一个数组
int x,y;
...
x=a[3];
y=b[3];
解释执行两条赋值语句时会发生什么?
ChinaUnix里面有一高人的解释如下:
这个问题不会产生编译错误和链接错误,也就是说程序是合法的。当然运行时是会出问题的。指针本身的标识需要占用一个int长的空间;数组本身的标识是不占地方的,把数组第一个元素的地址作为数组的标识。所以这个问题有趣的地方在这里。
编译时
A里面编译器认为a是数组,b是指针,这是很直接的。
因为B(b)被认为是数组,所以运行的时候那个地址应该是一个数组的首元素地址,B(b[0]) = A(b),所以B(b)的第一个元素应该保存的是a的地址。
也就是说,如果A(a) = {n,......};,那么B(a)=n,B(b)=&A(b)=&A(a)
链接之后,b[3]是随机的,但是b[0]=&a和a=n是确定的。
有上图可知:
1、定义为数组声明为指针,将得到指针变量保存将以数组元素
2、定义为指针声明为数组,数组元素保存指针值
复习一下指针和数组的区别:
1、指针保存的是数据的地址,自身的地址由编译器分配
间接访问,先取得指针的内容,再加上取值符号
2、数组保存的是数据
直接访问,数组名代表首元素首地址,&数组名才是数组首地址