在理解二维数组前,首先回顾一下一维数组的知识 问题1: int a[5] ={1,2,3,4,5}; 该数组的首元素,第二个元素地址怎么表示? 验证答案: - #include <stdio.h>
- int main(int argc, const char * argv[]) {
- int a[5] = {1,2,3,4,5};
- printf("a = %p\n", a);
- printf("&a = %p\n",&a);
- printf("&a[0] = %p\n",&a[0]);
-
- printf("a + 1 = %p\n", a+1);
- printf("&a[0]+1 = %p\n",&a[0]+1);
- printf("&a[1] = %p\n",&a[1]);
- return 0;
- }
- a = 0x7fff5fbff790
- &a = 0x7fff5fbff790
- &a[0] = 0x7fff5fbff790
- a + 1 = 0x7fff5fbff794
- &a[1] = 0x7fff5fbff794
- &a[0]+1 = 0x7fff5fbff794
复制代码
问题二:
怎么访问该数组的首元素,第二个元素? - #include <stdio.h>
- int main(int argc, const char * argv[]) {
- int a[5] = {1,2,3,4,5};
- printf("a[0] =%d\n",*a);
- printf("a[0] =%d\n",a[0]);
- printf("a[1] =%d\n",*(a+1));
- printf("a[1] =%d\n",a[1]);
- return 0;
- }
- a[0] =1
- a[0] =1
- a[1] =2
- a[1] =2
复制代码
OK,请大家记住这两道题目 大家可以看到,在第一道题目中,a,&a,&a[0]是一样的,都是一个地址。并且a+1和&a[1]地址也是一样的。在第二道题目中,我用了两种方式取得内存空间里面的内容,一个是通过*+地址,取得变量中的内容,另一个是通过变量名直接取得变量中的内容。
好,相信大家对上面的东西应该没有问题了。那么就来看下二维数组。 老师原话: 所谓多维数组就是二维和大于二维的数组,在C语言中并不直接支持多维数组,包括二维数组。多维数组的声明是使用一维数组的嵌套声明实现的。一个一维数组的每个元素又被声明为一维数组,从而构成二维数组,可以说二维数组是特殊的一维数组。 我的理解是:二维数组是一个特殊的一维数组,可以看做是一个一位数组,里面的每个元素都分别是一个一维数组。 那么现在就来分析一下下面的这个等式 *(a+i) =a+I = a = &a = &a[0]+i 我通过定义一个数组来验证说明 inta[2][3] = {{1,2,3},{4,5,6}}; 等式中的i代表行数,也就是数组中的2
首先看下运行结果 - #include <stdio.h>
- int main(int argc, const char * argv[]) {
- int a[2][3] = {{1,2,3},{4,5,6}};
- printf("*(a+1) =%p\n",*(a+1));
- printf("a+1 =%p\n", a+1);
- printf("a[1] =%p\n", a[1]);
- printf("&a[1] =%p\n",&a[1]);
- printf("&a[0]+1 =%p\n",&a[0]+1 );
- return 0;
- }
- *(a+1) =0x7fff5fbff78c
- a+1 =0x7fff5fbff78c
- a[1] =0x7fff5fbff78c
- &a[1] =0x7fff5fbff78c
- &a[0]+1 =0x7fff5fbff78c
复制代码
可以看到等式的确是成立的。那么就有疑问了 *(a+1)和a+1为什么相等呢? 通过问题,来解决。 问题1: 请问该二维数组地址是多少?用式子表示 首先第一种:数组名 //这个是大家都没问题的 那么第二种:&数组名 // 这个就有人会有疑问了。请看一维数组中得问题1,那么怎么解释呢。老师曾经说过,数组名可以代表数组地址。数组名是一个常量,那么常量会有地址吧,对它取地址,就可以得到这个常量的地址了。所以用取地址符可以取到整个数组的地址 第三种:通过开篇的一维数组可以知道,我们对一维数组的首元素取地址也是该一维数组的地址。而二维数组是一个特殊的一维数组,我对它的首元素取地址是不是也可以得到数组地址呢? 来分析拆解一下这个二维数组 inta[2][3] = {{1,2,3},{4,5,6}}; 我把它拆解为 int a[2] ={a[0],a[1]}; a[0] ={1,2,3}; a[1] ={4,5,6}; 因此我对a[2][3]这个数组的首元素a[0]取地址可以得到数组的地址既:&a[0] 第四种:因为a[0]也是一个数组,所以&a[0] = a[0]. 第五种:对二维数组的首元素取地址&a[0][0] //这个是大家也没什么问题的。 这是首地址取得五种方法 问题二:请问对于拆分的数组int a[2]={a[0],a[1]}; 对于第二个元素,你怎么取地址??(参考开篇一维数组) 第一种:直接用取地址符&a[1] //参考一维数组取第二元素的方法 第二种:通过第一问,可以知道&a[1] = a[1] 第三种:数组首元素的地址加1:a + 1 // 此处的a不是数组a的地址的意思,而是a[0]的地址,回看一下一维数组的问题一。明白了没有??一定要把二维数组变成一维数组来看。 第四、五种:接着第三种来看下,因为a = a[0] = &a[0],所以可以把a+1换为a[0] + 1或&a[0]+ 1 好来看下等式 *(a+i) =a+I = a = &a = &a[0]+i 是不是后面的四个等式都已经解决了 好吧,不要捉急,*(a+i)=a+i马上就出来了 问题三(第二问续):你怎么取出拆分数组int a[2] ={a[0],a[1]}; 的第二个元素? 参考一下一维数组的第二问:是不是可以用*(a+1)来取出?此处的a指的是数组int a[2][3]的地址。那么问:*(a+1) 和a[1]相等吗?答案是肯定的,因为你就是用指针取出的a[1]。那么a[1]到底是什么呢?a[1]是拆分的数组,就是代表数组名,也就是地址! 好了,那么是不是*(a+1)=a[1]就出来了,又因为a[1] = a + 1 所以*(a+i) = a+i = a = &a =&a[0]+i 注意:一定不要把*(a+i) = a+i中的a当做是一个a,*(a+i)中的a指的时二维数组的数组,代表的时数组的地址!而a+i中的a指的是拆分过后数组a[0]的地址。
此文仅代表本人个人见解!如有冲突,敬请谅解,仅供参考,每个人都有不同的理解思路,加油!!!
|