感觉这个问题是C语言面试挺典型的一道问题,这么多年了一直能见到,拿过来分享给大家
#include <stdio.h>
int main ()
{
int a[5] = {1,2,3,4,5};
int *p = (int*)(&a + 1);//&a表示整个数组的地址
printf("%d %d" , *(a + 1), *(p - 1));
}
//输出结果为:2,5
*(a+1)
其实很简单就是指a[1]
,输出为2
.
问题关键就在于第二个点,*(p-1)
输出为多少?
解释如下
&a+1
不是首地址+1
,系统会认为加了一个整个a数组
,偏移了整个数组a
的大小(也就是5
个int
的大小)。所以int*ptr=(int*)(&a+1);
其实p
实际是&(a[5])
,也就是a+5
.
原因为何呢?
&a
是数组指针,其类型为int(*)[5];
而指针+1
要根据指针类型加上一定的值,不同类型的指针+1
之后增加的大小不同,a
是长度为5
的int
数组指针,所以要加5*sizeof(int)
,所以p
实际是a[5]
,但是p
与(&a + 1)
类型是不一样的,这点非常重要,所以p - 1
只会减去sizeof(int*)
,a
,&a
的地址是一样的,但意思就不一样了,a
是数组首地址,也就是a[0]
的地址,&a
是对象(数组)首地址,a+1
是数组下一元素的地址,即a[1]
,&a + 1
是下一个对象的地址,即a[5]
。