题目:
int a[10] = {1, 2, 3, 4, 5};
int *ptr = (int *)(&a + 1);
printf("%d, %d", *(a + 1), *(ptr - 1));
输出结果:
2, 5
是不是有点懵逼了?
解析:
其实就是我们要搞懂 a 和 &a 到底是啥?
a 是数组名, 也是数组的首个元素的指针
sizeof(a)=20, 此时a的类型为int[5]数组。
sizeof(a)=4,因为有取值符,表示把a当成一个指针(int*),而a指向数组的首地址,
即a=&(a[0]),即sizeof(a)=sizeof(&(a[0]))=sizeof(a[0])=sizeof(int)=4。
(a+1)中把a当成一个指针,a+1=a+sizeof(int),a+1指向a的下一个整形地址既&a[1]。因此(a+1)=*(&a[1])=a[1]=2。
(&a + 1)先取变量a的地址,并根据a的地址获得下一个与a同类型的相邻地址。根据前面所说的a的类型为int[5]数组。
(int)(&a + 1)把这个相邻地址显式类型转换为int类型的地址int ptr = (int)(&a + 1); &a+1=&a+sizeof(5int),因此&a+1指向的地址为&a[5]. (就是超出第五个数a[4]的下一个地址);
(int)(&a + 1)把这个相邻地址显式类型转换为int类型的地址int ptr = (int)(&a + 1); 所以ptr指向&a[5],并且ptr是一个int类型的指针。ptr-1=ptr-sizeof(int),故ptr-1指向&a[4] (第五个数)。因此,(ptr-1)的值即为a[4]=5。