数组名和指针的区别(数组名不是指针,是数组首地址)

chengdongyuan 2017-02-02

听了一堂C语言的课,那老师说:“数组名就是一个指向数组首地址的常量指针”。我上百度查了一些,有好多教程、书籍等,都持相同的观点。但我一直感觉——数组名不等于指针。

实践是检验真理的唯一标准,于此,有了以下内容。

首先,声明一个数组和一个常量指针并指向那个数组。

1     int arr[3] = { 1, 2, 3 };      //声明数组
2     const int *p_arr = arr;        //声明常量指针

设问:一个整型指针的长度为4Byte,比较数组名的数据长度和指针的数据长度是否相等?

printf("*p_arr的长度=%d,arr的长度=%d\n", sizeof(p_arr),sizeof(arr));

运行结果:p_arr的长度=4,arr的长度=12

结论:数组名应该不是指针。

设问:数组名不能作为左值,那常量指针呢?

p_arr = p_arr;

运行结果:不报错

arr = arr;

运行结果:报错——“=”做操作数必须左值 and 表达式必须是可修改的左值。

const int *p_arr = &arr[0]; //重新声明一个,指向数组首地址 
printf("p_arr的地址=%p,arr的地址=%p\n", &p_arr, &arr);

运行结果:p_arr的地址=002DF958,arr的地址=002DF964。

结论:数组名不是常量指针,更不是指向数组首地址的常量指针。

设问:常量指针能为左值,那常量呢?

1     const int a = 3;
2     a = a;

运行结果:报错——“a”不能给常量赋值 and 表达式必须是可修改的左值。

结论:常量指针不是一个常量。

设问:数组名是一个常量吗?

printf("3的地址=%p,arr的地址=%p\n", &3, &arr);

运行结果:报错——常量上的“&” and 表达式必须为左值或者函数提示符。(数组名存在地址,而常量不存在)

#define B 5
printf("B的地址=%p,arr的地址=%p\n", &B, &arr);

运行结果:报错——常量上的“&” and 表达式必须为左值或者函数提示符。(与上一致)

printf("a的地址=%p,arr的地址=%p\n", &a, &arr); //a是之前用const声明过的常量

运行结果:能正常运行,说明a和arr都有自己的存储空间。

1     a = arr[0]; //因为arr=arr[0],现在那arr[0]的值赋给a
2     printf("a的地址=%p,arr的地址=%p\n", &a, &arr);

运行结果:a的地址=0041F7FC,arr的地址=0041F814(二者不在同一块内存单元中)

结论:数组名不是一个常量,也不是一个用const限定变量所形成的常量。

设问:数组名是否完成等同于数组首地址?

&arr[0] = &arr[0];

运行结果:报错——“=”做操作数必须左值 and 表达式必须是可修改的左值。(和arr = arr;的结果一致)

printf("arr[0]的地址=%p,arr的地址=%p\n", &arr[0], &arr);

运行结果:arr[0]的地址=002FFA34,arr的地址=002FFA34。(地址完全相等)

printf("&arr的长度=%d,arr的长度=%d\n", sizeof(&arr[0]), sizeof(arr));

运行结果:&arr的长度=4,arr的长度=12。

结论:数组名应该就是数组首地址,只是当在使用sizeof运算符时,可能进行了隐式转换。(转换指代整个数组,既sizeof(arr)=数组中元素的个数 乘以 sizeof(arr[0])的值)

最后的结论:暂且认为数组名在一般情况下就是该数组的首地址,在使用sizeof运算符时,数组名指代自身数组的全部元素的地址。

相关推荐