本章以二维数组为例介绍多维数组的指针变量。
1.多维数组的地址
设有整型二维数组a[3][4]如下:
0 1 2 3
4 5 6 7
8 9 10 11
它滴定义为:int a[3][4] = {{0,1,2,3},{4,5,6,7},{8,9,10,11}}
设数组a的首地址为1000,各下标变量的首地址及其值如图所示。
1000 0 | 1002 1 | 1004 2 | 1006 3 |
1008 4 | 1010 5 | 1012 6 | 1014 7 |
1016 8 | 1018 9 | 1020 11 | 1022 12 |
前面介绍过,C语言允许把一个二维数组分解为多个一维数组来处理。因此数组a可分解为三个一维数组,即a[0],a[1],a[2]。每一个一维数组又含有四个元素。
例如a[0]数组,含有a[0][0],a[0][1],a[0][3]四个元素。
数组及数组元素的地址表示如下:
从二维数组的角度来看,a是二维数组名,a代表整个二维数组的首地址,也是二维数组0行的首地址,等于1000。a+1代表第一行的首地址,等于1008。
a[0]是第一个一维数组的数组名和首地址,因此也为1000。*(a+0)或*a是与a[0]等效的,它表示一维数组a[0] 0号元素的首地址,也为1000。&a[0][0]是二维数组a的0行0列元素首地址,同样是1000。因此,a,a[0],*(a+0),&a[0][0]是等同的。
同理,a+1是二维数组1行的首地址,等于1008。a[1]是第二个一维数组的数组名和首地址,因此也为1008。&a[1][0]是二维数组a的1行0列元素地址,也是1008。因此a+1,a[1],*(a+1),&a[1][0]是等同的。
由此可得出:a+i,a[i],*(a+i),&a[i][0]是等同的。
此外,&a[i]和a[i]也是等同的。因为在二维数组中不能把&a[i]理解为元素a[i]的地址,不存在元素a[i]。C语言规定,它是一种地址计算方法,表示数组a第i行首地址。由此,我们得出:a[i],&a[i],*(a+i)和a+i也都是等同的。
另外,a[0]也可以看成是a[0]+0,是一维数组a[0]的0号元素的首地址,而a[0]+1则是a[0]的1号元素首地址,由此可得出a[i]+j则是一维数组a[i]的j号元素首地址,它等于&a[i][j];
由a[i]=*(a+i)得a[i]+j=*(a+i)+j。由于*(a+i)+j是二维数组a的i行j列元素的首地址,所以,该元素的值等于*(*(a+i)+j)。
例10.22
#include <stdio.h>
int main()
{
int a[3][4] = {0,1,2,3,4,5,6,7,8,9,10,11};
printf("%d",a);
printf("%d",*a);
printf("%d",a[0]);
printf("%d",&a[0]);
printf("%d\n",&a[0][0]);
printf("%d",a+1);
printf("%d",*(a+1));
printf("%d",a[1]);
printf("%d",&a[1]);
printf("%d\n",&a[1][0]);
printf("%d",a+2);
printf("%d",*(a+2));
printf("%d",a[2]);
printf("%d\n",&a[2][0]);
printf("%d",a[1]+1);
printf("%d\n",*(a+1)+1);
printf("%d,%d\n",*(a[1]+1),*(*(a+1)+1));
return 0;
}
运行结果:
基础问题:例:printf(“%d”,a);这个printf();里的为什么可以这样用?
带着不解和疑惑我去搜了一下printf的标准格式
int printf (const char *format, ...);
a是一个数组名,数组名是啥?数组名本质是一个地址,地址又是啥?卧槽我问题好多,先不管。
a是一个包含了12个整型元素的二维数组名。数组名是数组的首地址,就是数组中的第一个元素的地址,是常量。
ok,所以它是常量,还是一个整数型的常量吗?让我看看是不是
是的,我一搜我就看出来了,它是常量的整数型地址。所以也就是为什么它输出的是一个-的地址。为什么是一个-的地址?不知道。
神奇的一幕:请看一下代码
#include <stdio.h>
int main()
{
int a[3][4] = {0,1,2,3,4,5,6,7,8,9,10,11};
printf("%d ",a);
printf("%d ",*a);
printf("%d ",a[0]);
printf("%d ",&a[0]);
printf("%d \n",&a[0][0]);
printf("%d ",a+1);
printf("%d ",*(a+1));
printf("%d ",a[1]);
printf("%d ",&a[1]);
printf("%d \n",&a[1][0]);
printf("%d ",a+2);
printf("%d ",*(a+2));
printf("%d ",a[2]);
printf("%d \n",&a[2][0]);
printf("%d ",a[1]+1);
printf("%d \n",*(a+1)+1);
printf("%d,%d \n",*(a[1]+1),*(*(a+1)+1));
return 0;
}
我把每个printf();里的%d后面都加了个空格(因为觉得不好看) ,然后输出的地址就全为正了,好神奇,不应该啊,对了,应该是随机值。我再打印一波看看。
果然,当我打印了两次之后,结果出现了-的值,所以这个是正常的
ok,今日学习完毕,芜湖~。