二维数组与二级指针数组
二维数组:数组的数组spa
二维数组的初始化3d
int a[3][2]={(1,2),(3,4),(5,6)}; printf("a[0][0]=%d\n",a[0][0]);
不少人以为很简单,很快告诉我答案是1,不过很惋惜错了,答案是2,认真看,花括号里面嵌套的是小括号不是花括号,因此就至关于int a[3][2]={2,4,6};指针
二维数组在内存中的储存方式code
int a[3][4]blog
二维数组的地址内存
&a、&a[0]、a、*aclass
int a[2][3]={1,2,3,4,5,6}; printf("a_size=%d\n",sizeof(a)); printf("a[0]_size=%d\n",sizeof(a[0])); printf("a[0][0]_size=%d\n",sizeof(a[0][0])); printf("&a=%p\n",&a); printf("&a+1=%p\n",&a+1); printf("&a[0]=%p\n",&a[0]); printf("&a[0]+1=%p\n",&a[0]+1); printf("a=%p\n",a); printf("a+1=%p\n",a+1); printf("&a[0][0]=%p\n",&a[0][0]); printf("&a[0][0]+1=%p\n",&a[0][0]+1);
&a、&a[0]、a、&a[0][0]它们的数值上是相等,可是表示的意义各不相同 引用
&a:整个二维数组的首地址,&a+1,加的是sizeof(a)个字节。im
&a[0]:二维数组中大数组的首地址,何为大数组,就是a[0]、a[1],因此&a[0]+1,加的是sizeof(a[0])个字节。
a:本来觉得a是二维数组首元素的首地址,可是错了,它也表示的是二维数组中大数组的首地址,因此a+1,也是加了sizeof(a[0])个字节,把二维数组看成一维数组,只是一维数组中的元素也是一个数组,这样在二维数组中a表示二维数组中大数组的首地址就好理解了。
&a[0][0]:二维数组中首元素的地址,&a[0][0]+1,加的是sizeof(a[0][0])个字节。
经过指针操做二维数组
int a[2][3]={1,2,3,4,5,6}; int *p=a[0]; //int *p=a; int *p=&a; 报警告,数组名至关于二级指针,p为一级指针。 printf("a[0][0]=%d\n",*p); printf("a[0][1]=%d\n",*(p+1)); printf("a[1][0]=%d\n",*(p+3));
经过数组名来操做数组
int a[2][3];
&a[0]表明的是二维数组大数组的首地址,若是我要打印a[i][j],则我先要找到a[i]的首地址,&a[i]=&a[0]+sizeof(a[0])*i,既&a[i]=&a[0]+sizeof(int)*3*i,找到了a[i]的首地址,再来考虑a[i]中的内容,a[i][j]的地址为:&a[i]+sizeof(int)*j,既&a[i][j]=&a[0]+sizeof(int)*3*i+sizeof(int)*j;换算为指针形式表示为*(*(a+i)+j)
int a[2][3]={1,2,3,4,5,6}; printf("a[0][0]=%d\n",**a); printf("a[1][1]=%d\n",*(*(a+1)+1));
这时想到一个问题,上面代码中的*a与*(a+1),表明什么意思?
int a[2][3]={1,2,3,4,5,6}; printf("&a[0]=%p\n",&a[0]); printf("a=%p\n",a); printf("*a=%p\n",*a); printf("&a[1][0]=%p\n",&a[1][0]); printf("*(a+1)=%p\n",*(a+1));
经过查阅资料,发现虽然a与*a的值同样。可是它们的意义不一样,a 是数组地址,类型为int (*)[3],*a是元素地址,类型为int *,a比*a多一级解除引用。