对于数组名和对数组名取地址

本文深入探讨了C语言中数组名与取地址运算符的行为差异,解释了数组名、数组地址以及指针算术的基本概念,并通过实例展示了不同类型指针的步长计算方式。

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

对于数组名和对数组名取地址

前些日子看了一个帖子,其中有个例子如下:

 int a[5]={1,2,3,4,5};
 int *p1=(int*)(&a+1);
 int *p2=(int*)((int)a+1);
 int *p3=(int*)(&a)+1;
    // p3=p3+1;
 printf("%x,%x,%x",p1[-1],*p2,p3[-1]);

执行结果为5,2000000,1。对于结果5还是比较纠结。看了一下网上的帖子和几本书,总结了一下。

 

首先说一下关于对数组名取地址:
        关于对数组名取地址的问题,由于数组名是右值,本来&array 是不合法的,早期不少编译器就是指定&array 是非法的,但后来C89/C99认为数组符合对象的语义,对一个对象取地址是合理的,因此,从维护对象的完整性出发,也允许&array 。只不过,&array 的意义并非对一个数组名取地址,而是对一个数组对象取地址,也正因为如此,array 才跟&array 所代表的地址值一样,同时sizeof(array )应该跟sizeof(&array )一样,因为sizeof(&array )代表取一个数组对象的长度。

         要注意到 array 和 &array 的类型是不同的。array为一个指针,而&array是指向数组int [100]的指针。array 相当于 &array[0],而 &array 是一个指向 int[100] 的指针,类型是 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     //+sizeof(int)
&array=0012FDF0, &array+1=0012FF80  //+sizeof(&array)

在《C专家编程》书中关于数组一章P203,有如下解释:
    无论指针还是数组,在连续的内存地址上移动时,编译器都必须计算每次前进的步长。
    编译器自动把下标值调整到数组元素大小,对起始地址进行加法操作之前,编译器都会负责计算每次增加的步长,这就是为什么指针类型总是有类型限制,每个指针只能指向一种类型的原因所在,因为编译器需要知道对指针进行解除引用操作时应该取几个字节,以及每个小标的步长应取几个字节。
    另外步长的自动调整还和上下语句相关:
 int *p3=(int*)(&a);
      p3=p3+1;
    首先对P3指针变量赋初值,指向数组int [5]的指针,然后对指针进行加一的操作,其中P3定义为一个指向int类型的指针,因此最终P3的值等价P3+sizeof(int)
 int *p3=(int*)(&array+1);
    &array+1,步长为1,其中步长的长度和&array的类型匹配,即&array是指向数组int [100]的指针,所以&array+1等价为&array+sizeof(&array)
 
  最终p1[-1]等价为*(P1-1),因此等价为第二个int [5]的数据首地址(并不存在第二个数组显然当前指针已经越界了,另外数组元素在内存中是连续存贮的)减去一个为sizeof(int)的步长,所以指向了第一个数组的最后1个元素。

### 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、付费专栏及课程。

余额充值