(转+原创)c语言那些细节之a+1和&a+1的区别 ,指针的步长问题。

本文详细解析了C语言中数组名与数组首地址指针的区别,通过实例展示了两者在内存中的表现,并解释了如何使用数组指针进行元素访问。

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

首先a是一个数组名,当看到这个a&a时,一般我们的理解都是这个数组的首地址。没错,如果加上打印的话,确实两个值是一样的。

不过&a是整个数组的首地址,a则是数组首元素的地址,虽然值一样,但是意义却不相同。

在此之前有必要先看下c程序在内存中的分布图。其中,栈是向低地址位扩展的。



下面来看一个例子吧还是。

示例代码:

#include <stdio.h>
int main(int argc, char *argv[])
{
	int i;
	int a[]={1, 2, 3, 4, 5};
	int s;
	int *p = (int *)(&a+1);
	printf(" a = %p\n&a = %p\n", a, &a);
	for(i = 0; i < 5; i++)
		printf("a[%d] = %p\n", i, &a[i]);
	printf(" p = %p\n&p = %p\n&s = %p\n", p, &p, &s);
	return 0;
}

运行结果:(关于地址,这里也要考虑到内存地址对齐问题(2大原则),不同编译器处理方式不同)

a = 0022FF40
&a = 0022FF40
a[0] = 0022FF40
a[1] = 0022FF44
a[2] = 0022FF48
a[3] = 0022FF4C
a[4] = 0022FF50
 p = 0022FF54
&p = 0022FF38
&s = 0022FF3C

由上面的运行结果,我们可以知道a&a的区别了。可得下图:



因为是运行在main函数体内的变量,所以这些都是在栈中运行的,所以p指针是指向了a[4]后的那个地址,而p本身的地址是在栈中分配的。

综上可知:

&a+i = a + i*sizeof(a);  

a+i = a +i*sizeof(a[0]);   

本文转自:http://blog.youkuaiyun.com/eastmoon502136/article/details/8630790

 

已经比较了前两种,那么看下,ptr3的值是多少。

     int a[5] = {1, 2, 3, 4, 5};

     int *ptr1 = (int*)(&a + 1); //&a指针指向的元素为整个数组,故加为sizeof(数组)
     int *ptr2 = (int*)(a + 1); //a为数组首元素的地址,a+1为数组第二个元素的地址
     int *ptr3 = (int*)((int)a + 1); //a地址再加一个字节,直接地址值相加而不是指针
     int *ptr4=(int*)((int)a );
运行结果为: ptr4= 0012FF4C
                      ptr3=0012FF4D //仅仅是值加1
 
 
 
3.二级指针 一级指针存放普通变量的地址 二级指针存放一级指针的地址 格式: 存储类型 数据类型 **指针变量名 int **p; int a=333; int *p=&amp;a; int **q=&amp;p; q = &amp;p *q = *&amp;p = p = &amp;a **q = *p = *&amp;a = a 取变量a的值: a *p **q 取变量a的地址: &amp;a p *q 取p的地址: &amp;p q 4.指针数组 直接访问:通过数组名访问 间接访问:通过指针访问 直接访问: int a[5]={5,6,7,8,9}; 地址 元素 a &amp;a[0] 5 a[0] *a a+1 &amp;a[1] 6 a[1] *(a+1) a+2 &amp;a[2] 7 a[2] *(a+2) a+3 &amp;a[3] 8 a[3] *(a+3) a+4 &amp;a[4] 9 a[4] *(a+4) 间接访问: int a[5]={5,6,7,8,9}; int *p=a; 地址 元素 &amp;p[0] p 5 *p p[0] &amp;p[1] p+1 6 *(p+1) p[1] &amp;p[2] p+2 7 *(p+2) p[2] &amp;p[3] p+3 8 *(p+3) p[3] &amp;p[4] p+4 9 *(p+4) p[4] ap一样吗: 不一样,a是地址常量,不能被修改,p可以 直接访问: 元素:*(a+i) a[i] 地址:a+i &amp;a[i] 间接访问: 元素:p[i] *(p+i) 地址:p+i &amp;p[i] 补充: char a[]="hello"; // 栈区开辟6字节存放"hello" char *p="world"; //栈区开辟8字节存放 "world"所在常量区的首地址 指针二维数组 直接访问: int a[2][3]={4,5,6,7,8,9}; 二维数组的数组名是行地址,行地址>列地址 加*降级成了列地址 a : 首行地址 a+1: 第二行首地址 *a: 第一行第一列 *a+1: 第一行第二列 地址 元素 *a a a[0] &amp;a[0][0] 4 a[0][0] *a[0] **a *a+1 a[0]+1 &amp;a[0][1] 5 a[0][1] *(a[0]+1) *(*a+1) *a+2 a[0]+2 &amp;a[0][2] 6 a[0][2] *(a[0]+2) *(*a+2) *(a+1) a+1 a[1] &amp;a[1][0] 7 a[1][0] *a[1] **(a+1) *(a+1)+1 a[1]+1 &amp;a[1][1] 8 a[1][1] *(a[1]+1) *(*(a+1)+1) *(a+1)+2 a[1]+2 &amp;a[1][2] 9 a[1][2] *(a[1]+2) *(*(a+1)+2) 间接访问: int a[2][3]={4,5,6,7,8,9}; int *p=a; 直接访问: 元素:a[i][j] *(a[i]+j) *(*(a+i)+j) 地址:*(a+i)+j a[i]+j &amp;a[i][j] 用Markdown形式写一篇笔记
最新发布
04-01
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值