涉及数组成员的取址,可以使用数组名加偏移量(p = arr+5)或数组加下标再取地址(p = &arr[5]),那&arr代表什么呢?我一开始只知道arr和&arr的值相等。
#include <stdio.h>
int arr[10];
int main(int argc, char *argv[])
{
printf("%p ,%p, %p\n", &arr, arr, &arr[0]);
return 0;
}
/*
Output:
00407020 ,00407020, 00407020
*/
但是其实&arr和arr的意义不同。对于以上代码,&arr返回一个int(*)[10]的数组指针,而arr是int*型。
如果有如下函数:
void IntPointerTest(int* p)
{
(void)p;
}
当调用IntPointerTest(&arr)时,就会报错(这里是C++编译器,C编译器可能只报Warning):
C++编译器:
[Error] error: cannot convert ‘int (*)[10]’ to ‘int*’ for argument ‘1’ to ‘void IntPointerTest(int*)’
C编译器:
[Warning] note: expected ‘int *’ but argument is of type ‘int (*)[10]’
数组在一定程度上和结构体类似,int arr[10]可以看作是:
struct arr_tag
{
int _[10];
} st_arr;
sizeof(arr)相当于sizeof(st_arr),为40,对应的&arr+1相当于&st_arr+1,与&arr相比,地址偏移了40。
引用iteye_10992的博客:
关于对数组名取地址的问题,由于数组名是右值,本来&array 是不合法的,早期不少编译器就是指定&array 是非法的,但后来C89/C99认为数组符合对象的语义,对一个对象取地址是合理的,因此,从维护对象的完整性出发,也允许&array 。只不过,&array 的意义并非对一个数组名取地址,而是对一个数组对象取地址,也正因为如此,array 才跟&array 所代表的地址值一样,同时sizeof(array )应该跟sizeof(&array )一样,因为sizeof(&array )代表取一个数组对象的长度。