请先看以下程序:
Q:
若 int a[5];
则 &a+1 的含义是甚么?
A:
&a+1 不是表示a的地址(设为Ox0010)加1,变为0x0011. 由于a为包括5个int类型的数组,则"&a+1"中的"+1"表示为相当于"1"个a大小的空间(或成为偏移),此时&a+1 表示 a[5].
Q:
若 int* ptr = (int*) (&a+1);
则 ptr 含义是甚么 ?
而 ptr - 1 又表示甚么 ?
A:
由于&a+1 表示 a[5], 则ptr即为a[5]。
又 ptr 为int型的指针,故 "ptr-1"则会减去"1"个int型指针的空间,此时即为a[5-1]=a[4].
看下面具体程序:
main()
{
int a[5]={1,2,3,4,5};
int *ptr=(int *)(&a+1);
printf("%d,%d",*(a+1),*(ptr-1));
}
输出:2,5
Explanation:
*(a+1)就是a[1],*(ptr-1)就是a[4],执行结果是2,5
&a+1不是首地址+1,系统会认为加一个a数组的偏移,是偏移了一个数组的大小(本例是5个int)
int *ptr=(int *)(&a+1);
则ptr实际是&(a[5]),也就是a+5
原因如下:
&a是数组指针,其类型为 int (*)[5];
而指针加1要根据指针类型加上一定的值,
不同类型的指针+1之后增加的大小不同
a是长度为5的int数组指针,所以要加 5*sizeof(int)
所以ptr实际是a[5]
但是prt与(&a+1)类型是不一样的(这点很重要)
所以prt-1只会减去sizeof(int*)
a,&a的地址是一样的,但意思不一样,a是数组首地址,也就是a[0]的地址,&a是对象(数组)首地址,a+1是数组下一元素的地址,即a[1],&a+1是下一个对象的地址,即a[5].
分析如下代码:
1
2
|
int
a[3] = {1,2,3};
printf
(
"%d\n"
, *((
int
*)(&a+1) - 1));
|
为什么?下面我们就来一一分析
大家应该都知道,a的地址和&a[0]的地址一样,那么和&a的地址也一样。a和&a[0]是等同的,但是a和&a表达的意思却是不一样的。 a和&a[0]的类型都是 int *,即指向int的指针。而&a的类型是int *[3],即指向3个int元素的数组的指针。&a 这是一个指向数组a[3]的地址,即a[3]的第一个元素所在的地址。编译器将数组的地址默认为数组第一个元素的实际地址,也就是数组的首地址。
&a 可以说是一个二级指针,那么(&a + 1)则移动的也是一个单元,但不是一个int单元,而是一个int[3]单位,就是整个数组都移动过去了。(&a + 1)也就是移动到a[3]的下一个单元位置了,在这里我们可以打印一下(&a + 1)的地址看一下,我们将会发现 (&a + 1)的地址和a的地址正好相差(sizeof(int) * 3)个字节的大小。
那么(int *)(&a + 1)是什么意思呢,为什么在(&a + 1)前面还要加上(int *)呢,很简单(int *)这是一个强制类型转换,把这个二维的指针再强制转换成一维的