一维数组的数组名是指向单个变量的指针。
二维数组的数组名是指向单个数组的指针(行指针)。
二维数组(如:a[3][4])的指针分为两种:
1.行指针,如:a,是一个指向数组的指针;
2.列指针,如:&a[0][0],a[0],*a,是一个指向单个变量的指针。
虽然a,&a[0][0],a[0],*a,在物理上表示都是同一块内存空间的地址,但在概念上它们的意义不同。
a表示的指针是指向整块数组的。
&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)+2))。
如果是列指针,则直接进行解引用即可(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* pa = &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 }