C/C++拾遗:关于数组的指针和数组元素首地址的一道经典题

本文探讨了C/C++编程中数组、指针和数组元素首地址的关系,强调了&a、a、&a[0]在内存表示上的等价性,并解释了在类型转换中可能产生的理解误区。

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

 #include <stdio.h>int main(void){
      int  a[5] = {1, 2, 3, 4, 5};   
      int  *ptr = (int *)(&a+1); 
      int *p1 = a;    
      int *p2 = &a[0];    
      int *p3 = (int *)(&a);  
      if(p1 == p2){     
        printf("p1 == p2\n");   
      }else{      
        printf("p1 != p2\n");   
      }if(p1 == p3){     
        printf("p1 == p3\n");   
      }else{      
        printf("p1 != p3\n");   
      }if(p2 == p3){      
        printf("p2 == p3\n");   
      }else{      
        printf("p2 != p3\n");
      }  
      printf("%d %d\n",*(a+1),*(ptr-1));  
      int *p4 = ++p1; 
      int *p5 = ++p3; 
      if(p4 == p5){     
        printf("p4 == p5\n");   
      }else{  
        printf("p4 != p5\n");   
      }   
       return 0;
 }  
最终输出结果如下:
里面所有的判断都是相等,打印的两个值是2和5.
原因如下:
&a是数组的首地址,它的类型是int(*)[5],因此+1是加的是数组个数的步长。指针加1要根据指针自身类型加上一定的值,
不同类型的指针+1之后增加的大小不同。因此&a + 1后指向的地址对数组来说是越界的,注意前面有个int(*)又强制将他转为int*了。
因此计算*(ptr - 1)的时候,ptr是个int类型的指针,减一等于往左移动sizeof(int*)个字节,因此又指向a[4]了,他的值等于&a[4].
为此大家又对p4和p5作了验证,为啥结果p4会等于p5呢?原因就在于申请p3的时候对&a进行了强制类型转换,
view plaincopy to clipboardprint?
int *p3 = (int *)(&a);  
如果不加这个转换,是编译不过去的。加了转换之后p3等于p1也等于p2,他的类型跟另外两个一样都是int*类型的了,

因此移动相同位后地址也是一样的。很多人纠结就纠结在&a、a、&a[0],根本原因在于对&a进行了强制类型(int*)转换。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值