数组名 和 数组名取地址 的不同

本文探讨了C语言中数组名与对数组取地址的区别,包括类型差异、sizeof运算符的应用及指针操作的不同表现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

    昨天晚上在cppblog上逛,发现一个关于对数组名取地址 的帖子,地址为http://www.cppblog.com/cuigang/archive/2008/04/04/46271.html
其中说到:



1 int array[100];
2 
3 memset(array,  0, sizeof(array));
4 memset(&array, 0, sizeof(array));

第3行和第4行有什么不同吗?其实从效果上来说是一样的,但是这里要注意 array 和 &array 的类型是不同的。array 相当于 &array[0],而 &array 是一个指向 int[100] 的指针,类型是 int(*)[100]。
以下代码可以看出这个不同:

 1 #include <stdio.h>
 2 
 3 int main()
 4 {
 5         int array[100= {012};
 6         typedef int (*ARRAY)[100];
 7         int* p1 = array;
 8         ARRAY p2 = &array;
 9         //int* p3 = &array;     //这样编译错误
10         
11         printf("p1 = 0x%08X/n", p1);
12         printf("p2 = 0x%08X/n", p2);
13         printf("p1[2] = %d/n", p1[2]);
14         printf("p2[2] = %d/n", p2[2]);
15         printf("(*p2)[2] = %d/n", (*p2)[2]);
16         
17         getchar();
18         return 0;
19 }

运行结果可能是:

p1 = 0x0022FDF8
p2 
= 0x0022FDF8
p1
[2] = 2
p2
[2] = 2294040
(*p2)
[2] = 2


   由于习惯使用sizeof来判断一个变量的类型,而实验了一下:
int array[100];
memset(array, 0, sizeof(array));

printf("sizeof(array)=%d, sizeof(&array)=%d/n", sizeof(array), sizeof(&array));

     发现sizeof(array)和 sizeof(&array)结果一样,都为400,所以想当然的认为这两个的类型是一样的。

     回头又查了下C专家编程(第9章),其中有说:
表达式中的数组名被编译器当作一个指向该数组第一个元素的指针。也即array为一指针类型。
另外注释中有提到:在使用sizeof()、&操作符取数组指针、数组为一字符串常量初始值的情况下,对数组的引用不能用指向该数组第一个元素的指针来代替。

   因此array和&array类型是不一样的。array为一个指针,而&array是指向数组int [100]的指针。下面的代码(计算步长)也可以证明这个结论:

printf("array=%p, array+1=%p/n", array, array+1);
printf("&array=%p, &array+1=%p/n", &array, &array+1);

结果为:
array=0012FDF0, array+1=0012FDF4
&array=0012FDF0, &array+1=0012FF80
### C/C++ 数组指针与数组名取地址的区别 #### 数组名的本质 在 C C++ 中,数组名本质上是一个指向数组首元素的常量指针。这意味着当提到数组名时,实际上是在提及该数组第一个元素的位置[^1]。 ```cpp int arr[5]; // 这里的arr表示的是数组的第一个元素(&arr[0])所在的内存位置 ``` 然而需要注意的是,尽管 `arr` 表现得像一个指针变量,但它并不是真正的指针——它不会被存储在一个可变的对象中,也不能改变其指向的内容。因此,在某些情况下将其视为指针可能会引起混淆或错误[^3]。 #### 取地址运算 (`&`) 应用于数组名的结果 如果对整个数组应用取地址运算,则会得到一个指向此数组类型的指针而不是单个元素的指针: ```cpp int (*pArr)[5] = &arr; // pArr 是一个指向含有五个整型元素的一维数组的指针 ``` 这表明 `&arr` 返回了一个不同类型的指针,即指向整个数组而非仅限于首个元素的指针。这种区别对于理解如何传递多维数组作为函数参数尤为重要。 #### 指向数组的指针 另一方面,可以通过声明显式的指针来创建能够遍历数组各个元素的真正意义上的指针对象: ```cpp int *ptr = arr; // 或者写作 int *ptr = &arr[0]; // ptr现在是指向int类型数据的一个普通指针 ``` 这样的指针可以在运行期间更改所指向的位置,并支持算术操作如自增(`ptr++`)以访问后续元素[^2]。 #### 使用场景对比 - **数组名**: 当只需要读或写入特定索引处的数据而不需要修改迭代器本身时适用;也可以用来初始化另一个相同大小类型的数组。 - **指向数组的指针**: 更灵活的选择,允许动态调整偏移量以及执行更复杂的逻辑处理,比如实现查找算法或是构建链表结构等。 综上所述,虽然两者看起来相似并且经常互换使用,但在实际编码过程中应当区分清楚它们之间的差异以免引入难以调试的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值