指针与数组
指针数组,数组指针,指针法访问数组,分不清如何使用,下面将帮你疏通思路,更好理解他们的关系。
数组名的本质
int array[5];
printf("array = %o\n",array);
printf("&array = %o\n",&array);
printf("&array[0] = %o\n",&array[0]);
代码结果
由结果可以得知,数组名是一个地址,指向数组的首地址。那样我们就可以通过指针来操作数组元素。
指针法操作数组元素
可以通过*array来操作数组的首元素,但如果想要操作第二个元素,要怎么办呢?地址也是可以进行运算操作的,通过改变地址的值来指向不同的内存,这样就有可能操作数组的第二个元素。
首先看看数组的元素地址
数组地址
可以看到数组元素地址是一段连续的地址,这样就有可能通过知道数组首地址,来操作数组的所有元素。上面的地址增量是4个字节,联想到该数组是int型,int型在32位平台占4个字节,所以这个增量是要看数组元素数据类型。
int array[5];
int *ptr = array;
printf("ptr = %o\n",ptr);
printf("ptr+1 = %o\n",ptr+1);
指针操作
可以看到ptr+1后地址开始由30177370变化为30177374,增量为4,这是为什么呢?这就有关指针步长了。指针步长指指针加1的长度增量。可以看到此时步长为4,4这个数字很熟悉,ptr是整型指针,整型在32位平台中占4个字节,和ptr指向的数组的元素地址增量一致。那么就可以通过指针来操作数组元素。所以对应类型的指针指向对应类型的数据,不然可能会出错。
*(ptr+1) = 2;
*(ptr+2) = 3;
也可以使用[ ]的形式操作, *与[ ]的效果一样,编译器内部做了处理。
ptr[1] = 2;
ptr[2] = 3;
过程
这种形式看起来ptr是一个数组名,阅读体验也不错。我一般是使用这种形式。但不能把ptr当作是一个数组,他本质还是一个指针。
ptr与array
指针步长在一维数组与二维数组
一维数组
int array[5] = {1,2,3,4,5};
printf("array: %u array+1: %u\n",array,array+1);
printf("&array: %u &array+1: %u\n",&array,&array+1);
printf("sizeof(array) = %d\n",sizeof(array));
指针步长
可以看到array与&array的指针步长不一样,array的步长为一个元素的字节数,&array的步长为一个数组的长度。
array是这个数组的首元素地址 +1的长度为一个数据类型长度
&array是这个数组的首地址 +1的长度为·一个数组类型长度
二维数组
int array[2][3] = {1,2,3,4,5,6};
printf("array[0]: %u array[0]+1: %u\n",array[0],array[0]+1);
printf("array: %u array+1: %u\n",array,array+1);
printf("&array: %u &array+1: %u\n",&array,&array+1);
printf("sizeof(array) = %d\n",sizeof(array));
printf("sizeof(array[0]) = %d\n",sizeof(array[0]));
二维数组指针步长
&array 这个二维数组的首地址 指针步长为整个二维数组
array 这个二维数组的首行地址,指针步长为每行的一维数组长度
*array 首行首元素地址,指针步长为元素的数据长度
*array 这个数组的首行首元素数据
这个对照array[ ]表示法就可以明白了,等同于[ ]。
*array == array[0] 就表示首行首元素地址。也可以是 &array[0][0],&与一个[ ]抵消。
**array == array[0][0] 这样就很好理解了
注意事项
数组名不可以为左值,数组名虽然本质是一个地址,但这个只可以读,不可以修改。
int array1[5];
int array[5];
int *ptr = array1;
array = ptr; //error,编译出错
指针与数组的简单介绍到此结束
网友评论