有两个我认为值得分析的指针例程,希望大家看了能对大家有帮助
eg1:
#include <stdio.h>
int main()
{
int a[4]={1,2,3,4};
int *p1 = (int *)(&a+1);
int * p2 = (int *)((int)a+1);
int * p3 = (int *)(a+1);
printf("%d,%d,%d\n",p1[-1],p2[0],p3[1]);
return 0;
}
这是第一个例程,编译之后,运行,会打印什么结果呢?
后面会给出答案,但这里希望大家停下来思考,再去看后面的答案.
分析:
p1:&a为整个数组a的地址,&a+1就直接指到数组a的最末端了,然后我们我们再(int *)(&a+1),这样就把指向数组最末端的指针强制转换成int型的指针,这样,后面的打印p1[-1]就是指针向前移动一个int型的距离,直接指到a数组的最后一个元素了,故打印出来是4;
p2:a为数组元素首地址,所以(int)a就是将首地址强制类型转换为int型,再加1,就是将指针指到从第一个元素地址的第二个字节开始取一个int型的数字,也就是0x0200 0000了.
p3:a为第一个元素的首地址,所以a+1就是第二个元素的首地址,所以p3指向a[1],p3[1]就是a[2]了,就是为3.
eg2:
#include <stdio.h>
void f(int a[10000])
{
printf("%d\n",sizeof(a));
}
int main()
{
int a[5]={0};
f(a);
return 0;
}
直接编译运行,这个打印会是什么呢?
这个打印4,我第一反应就是sizeof(a),应该打印4*10000=40000啊,可是我最后才知道编译器编译的时候,会将a[10000]替换为void f(int *a),这个例程很简单,因为我第一反应是出错了,所以我拿出来在这提醒下自己.