二维数组的指针

一维数组的数组名是指向单个变量的指针。

二维数组的数组名是指向单个数组的指针(行指针)。

二维数组(如:a[3][4])的指针分为两种:

1.行指针,如:a,是一个指向数组的指针;

2.列指针,如:&a[0][0],a[0],*a,是一个指向单个变量的指针。

虽然a,&a[0][0],a[0],*a,在物理上表示都是同一块内存空间的地址,但在概念上它们的意义不同。

a表示的指针是指向整块数组的。这时候我们把二维数组看作是几个一维数组组成的(如:a[3][4]是由三个一维数组组成的)。

&a[0][0],a[0],*a表示的指针是指向单个变量的。这时候我们把二维数组直接看做一维数组,元素像一维数组中一样一个挨一个排列。

对a进行解引用(即:*a)仍然会得到一个指针(并且它们的地址值相同……)。

所以可以推测出,对一个行指针进行解引用只是改变了其概念而已(而不是访问a中保存的地址所表示的内存空间),具体是指:原来它指向的是一个数组,解引用后指向这个数组中的第一个元素,如果再解引用你就可以访问这个元素了。

总结:

a,&a[0][0],a[0],*a表示的都是指针,并且是同一块内存的地址,但它们表示的意义是不同的。

a表示行指针(指向一个数组),*a,a[0],&a[0][0]表示列指针(这三个是等价的,都指向数组中的一个元素)。

以取a[2][1]的值为例:

如果是行指针,则先要对它进行解引用,变为一个列指针,然后在解引用得到其值(a[2][1]=*(*(a+2)+1))。

如果是列指针,则直接进行解引用即可(a[2][1]=*(*a+2*4+1))。

 

测试代码如下:(int类型占四个字节的情况下)

 1 #include<stdio.h>
 2 int main()
 3 {
 4     int a[3][4] = { 1,2,3,4,5,6,7,8,9,10,11,12 };
 5     //int* p1 = &a[0][0];
 6     //int *p1 = a[0];
 7     int* p1 = *a;    //列指针,指向一个单个变量
 8     int(*p2)[4] = a;    //行指针,指向一个含有四个元素的数组
 9     //以下代码表明:a,a[0],*a,&a[0][0]都是指针,并且里面的值也是一样的
10     printf("%p\n", a);
11     printf("%p\n", a[0]);
12     printf("%p\n", *a);
13     printf("%p\n", &a[0][0]);
14     putchar('\n');
15     //以下代码表明:a与*a,a[0],&a[0][0]的不同之处
16     printf("%p\n", a);
17     printf("%p\n", a + 1);    //对一个行指针加一,地址值向后移动了十六位(四个整形变量所占的内存)
18     printf("%p\n", *a);
19     printf("%p\n", *a + 1);    //对一个列指针加一,地址值向后移动了四位(一个整形变量所占的内存)   
20     printf("%p\n", a[0]);
21     printf("%p\n", a[0] + 1);
22     printf("%p\n", &a[0][0]);
23     printf("%p\n", &a[0][0] + 1);
24     putchar('\n');
25     //以下代码表明:对行指针进行解引用,得到的仍是一个指针,但它不再是一个行指针了(变成了列指针)
26     printf("%p\n", *a);
27     printf("%p\n", *a + 1);
28     putchar('\n');
29     //以下代码为取a[2][1]的值
30     printf("%d\n", a[2][1]);
31     printf("%d\n", *(*(a + 2) + 1));
32     printf("%d\n", *(*a + 2 * 4 + 1));
33 }
posted @ 2019-07-24 20:45  奔_gis  阅读(7990)  评论(0编辑  收藏  举报