给大家呈现一个程序:
#include<stdio.h>
void main()
{
int a[3][4]={{0,1,2,3},{4,5,6,7},{8,9,10,11}};
int *p=a[0];
for(;p<a[0]+12;p++)
{
if((p-a[0])%4==0) putchar('\n');
printf("%4d",*p);
}
putchar('\n');
}
这个程序我明白,p=a[0];这个语句说明了指针变量p指向0行0列元素,*p 就是元素值。
如果我改动一下程序:
#include<stdio.h>
void main()
{
int a[3][4]={{0,1,2,3},{4,5,6,7},{8,9,10,11}};
int *p=a;//这里改动了,让p由原来的列指针变为行指针;
for(;*p<a[0]+12;*p++)
{
if((*p-a[0])%4==0) putchar('\n');
printf("%4d",**p);//**p 就是元素值;
}
putchar('\n');
}
我的想法是:
令 指针变量 p=a; 即指针变量p是个行指针(原程序 p 是列指针),那么 *p 就是列指针(原程序 *p 是元素值),那么**p就是元素值。
根据这个想法我编出了第二个程序,但是编译出错。我找不到思路上哪里有问题,请懂的人指教一下。
数组在内存中是连续存储的,在你的改动处 int *p=a;p仍然指向数组的第一个元素,你必须明确在数组a[3][4]中,a、a[0]、a[0][0]的地址完全相同,而意义不同。a表示数组的首地址同时也是第一个元素的地址,a[0]表示数组中第一行的行首地址,a[0][0]表示数组的第一个元素的地址。你定义的int *p=a,实际上是指针p指向了数组的第一个元素的地址,这里p并不是一个行指针。
根据你的定义程序改写为:
a[3][4]在内存中式一点连续的线性空间
a[0][0]、a[0][1]、a[0][2]、a[0][3]、a[1][0]、a[1][0]、a[1][0]...
依次排列,内存分布详见插图。
所以并不存在你描述的行指针与列指针的关系
(int*)a a[0] (int*)&a[0] &a[0][0]指向同一个地址 int *p=(int*)a;
*p可以取得当前内存中的数据,但是**p应该是非法的无意义的。
int *p=(int*)a;
for(;p<a[0]+12;*p++)
{
if((p-a[0])%4==0) putchar('\n');
printf("%4d",*p);
}
putchar('\n');