对C的指针总是理解不够清晰,看书时又发现一遗漏之处,现补充理解上,
int a[5] = {1,2,3,4,5};
int *ptr = (int*)(&a+1);
printf("%d,%d\n",*(a+1),*(ptr-1));
printf("%x,%x\n",&a[0],&a[0]+1);
printf("%x,%x\n",&a,&a+1);
执行结果 2,5
19f900,19f904
19f900,19f914
(环境为VC++2008)
a和&a的值虽然一样,但是意思不一样,a是数组元素的首地址,也就是a[0]的首地址,&a是数组的首地址,a+1就是数组的下一个元素的地址即a[1]的地址,所以输出2,&a+1是下一个数组的首地址;(这条很重要,一定要区分开a和&a)
*(ptr-1)因为ptr是指向a[5]的,并且ptr的类型是int*,所以*(ptr-1)是指向a[4],输出5;
&a+1是取数组的首地址,该地址的值加上sizeof(a)的值,即&a+5*sizeof(int),也就是下一个数组的首地址,显然当前指针已经越过了数组界限;
(int*)(&a+1)是把上一步计算出的地址强制转换为int*赋值给ptr
理解:对指针进行加1操作,得到的是下一个元素的地址,而不是原有地址值加上1,所以一个类型为T的指针的移动,以sizeof(T)为移动单位。因此对于上面的code来说a是一个一维数组,数组中友5个元素,ptr是一个int型的指针。
另外说一句*的优先级比“+,-”高,所以*ptr+1,*(ptr+1)不能等同,提醒自己。