C语言中的数组名含义(何时代表整个数组,何时代表数组首地址,解索引取值)代码加图详解

数组名的含义
一、代表整个数组
  1. 定义数组
  2. 使用sizeof
  3. 使用&符号

代码示例

int a[3];					//a代表整个数组
printf("%lu\n",sizeof(a);	//a代表整个数组
printf("a的地址为:%p\n",&a);	//a代表整个数组
二、代表数组首地址
  • 不是上面三种情况时
int a [4] = {1,2,3,45};
int *p = a;		//a代表首元素地址---->&a[0]
p = a+1;		//a代表首元素地址---->&a[0]
func(a);		//a代表首元素地址---->&a[0]
scanf("%d\n",a);//a代表首元素地址---->&a[0]
..........
例题详解
  • 代码示例
	int a[5] = {1,2,3,4,5};
    int b[3][4] = { {1,2,3,4}, {5,6,7,8},{9,10,11,12}}; 

    printf("%p\n",&a); //数组a的地址      
    printf("%p\n",&a+1);//数组a地址+20
    printf("%p\n",a);  //a[0]地址          
    printf("%p\n",a+1); //a[1]地址
    printf("%p\n",a+2); //a[2]地址
    printf("%p\n",a+3); //a[3]地址 
    printf("%p\n",a+4); //a[4]地址  
        
//二维数组情况
    printf("\n");
    printf("%p\n",&b); //数组b的地址      
    printf("%p\n",&b+1); //3x4=12;int型4个字节-->12x4=48
                        //故地址为b+48(注意是十六进制的计算)   
    printf("\n");
    printf("%p\n",b); //b[0](也是b[0][0])的地址-->{1,2,3,4}     
    printf("%p\n",b+1); //b[1]地址 ;因为b[0]有4个元素,4x4=16;
                        //刚好一个16进制
    printf("\n");
    printf("%p\n",*b); //b[0][0] 地址     
    printf("%p\n",*b+1); //b[0][1]地址 
    printf("%p\n",*b+2);//b[0][2]地址  
    printf("\n");

    printf("%d\n",**b);//b[0][0]值     
    printf("%d\n",**b+1); //b[0][1]值

代码详解:

  1. &a+1代表的是整个数组加一,&a我们清楚它拿的是整个数组地址,这个地址是一个整体,你可以把它想象成一个木箱,至于木箱里面装了什么东西不用管,那么这个木箱+1,去到了木箱的末尾或者可以说是第二个木箱的头;所以&a+1其实拿到的是整个数组的长度,或者说是这个数组的末尾地址,如果它是一个二维数组,那你会发现&a+1的地址就是接着a后面的那个元素的首地址。

  2. &a的地址等于a的地址等于a[0]的地址,这里就有疑惑了,不是说&a代表整个数组,而a[0]是数组首地址,它两者怎么相等呢,这类比一个大木箱,里面整齐排列3个小木箱,那这个整个大木箱的头也是里面第一个小木箱的头,所以它们的地址是相等的,而里面3个小木箱的长度相加就是大木箱的长度(类似说法,方便理解,内存是不会像木箱一样有间隙的)。

  3. 那如何证明3小木箱长度相加就是大木箱的长度呢,我们把上面一段代码拿出来讲解
    代码示例

	int b[3][4] = { {1,2,3,4}, {5,6,7,8},{9,10,11,12}};
    printf("%p\n",&b); //数组b的地址      
    printf("%p\n",&b+1); //3x4=12;int型4个字节-->12x4=48
                        //故地址为b+48(注意是十六进制的

首先b这个数组是个int型二维数组,那每个元素的大小是4个字节,里面16个元素,4x16=48位,意味着&b+1的地址为&b的地址加个48;那我们来打印看下输出结果;

printf("%p\n",&b);  //0x7ffd910b34f0
printf("%p\n",&b+1);//0x7ffd910b3520

通过结果,我们发现十六进制 3520 比 34f0 大了3个十六进制,也就是48。(f---->2:中间经历着0,1,2三位;实在不懂进制计算的同学可以百度进制转换)

  1. 二维数组的*p和**p的取值问题

我们把前面的代码拿下来:

	int b[3][4] = { {1,2,3,4}, {5,6,7,8},{9,10,11,12}}; 
 	printf("%p\n",*b); //b[0][0]地址     
    printf("%p\n",*b+1); //b[0][1]地址 
    printf("%p\n",*b+2);//b[0][2]地址  
    printf("\n");

    printf("%d\n",**b);//b[0][0]值     
    printf("%d\n",**b+1); //b[0][0]值加1
  • 前面我们说过,&b是拿到b的整个地址,那*p解一层指向了里面的第一个元素{1,2,3,4}的首地址,*p+1则是移动到了第二个元素2的地址,以此类推,他们在二维数组中的表示方法为b[0][1].
  • *p是解一层拿到首元素{1,2,3,4}的首地址,那再解一层**p就通过地址拿到了里面的数值,再附上图片详解。
  • 图例详解
    在这里插入图片描述

到此,我们的讲解就结束了,其他内容大家自己查看打印结果,再根据我的注释加以思考其实是不难的,多思考多验证自己的想法,对于知识点的掌握和理解是有非常大的帮助的。

  • 12
    点赞
  • 59
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值