直接上代码,然后解释
#include <stdio.h>
int main()
{
printf(" 得到数组的最后一个数的趣味实现 \n");
printf(" - http://blog.youkuaiyun.com/morewindows/article/details/10022147 -\n");
printf(" -- By MoreWindows( http://blog.youkuaiyun.com/MoreWindows ) --\n\n");
const int MAXN = 8;
int a[MAXN] = {1, 2, 4, 8, 16, 32, 64, 128};
int *p = (int*)(&a + 1); //&a 是一个指针,指向大小为4的数组 int(*)[MAXN]
printf("%d\n", *(p - 1)); //为128
return 0;
}
地址:http://blog.youkuaiyun.com/morewindows/article/details/10022147
这是因为指针的加法和减法不同于其它数值的加减法,它是以sizeof(point)为单位的。
&a 是一个指针,指向大小为4的数组 int(*)[MAXN]
但是我觉得,&a 是一个指针,指向大小为8的数组 int(*)[MAXN]
如果这么定义:
int *p=(int *)(a+1);这样得到的*(p-1)就是第一个元素。
其实a已经是地址了,但是再次取值,这个时候跨度就是4个整数的跨度,指针的加减都是根据自己的类型加减的。如果数组定义为二维:
int a[MAXN][MAXN],如果此时a+1,地址就变为第二行的第一个元素。
具体的测试可以通过下面的代码验证:
#include <stdio.h>
#include <stdlib.h>
int main()
{
int i,j;
int b[5]={1,2,3,4,5};
int *p=(int*)(&b+1);
for(i=0;i<5;i++)
printf("%0x %d\n",&b[i],&b[i]);
printf("%0x %d\n",*p,*(p-1));
return 0;
}
从打印的结果就可以发现问题
bfe72e4c -1075368372
bfe72e50 -1075368368
bfe72e54 -1075368364
bfe72e58 -1075368360
bfe72e5c -1075368356
bfe72e60 5
下面这个还有一个实现,不过下面的好理解:
#include <stdio.h>
#include <stdlib.h>
int main()
{
const int max=5;
int a[5]={1,2,3,4,5};
printf("%d\n",a[sizeof(a)/sizeof(a[0])-1]);
return 0;
}