在讲解之前,我们得先了解,数组指针和指针数组之间的区别,在不了解其本质之前,很容易将二者搞混。其实区分方式很简单,我们只需要记住这两者的后缀就行,前面的词相当于是修饰作用。如下:
数组指针:指针、数组是修饰 存放整个数组的指针
指针数组:数组、指针是修饰 存放指针的数组
注意我们数组指针存放的值为整个数组的地址,而不是数组的首元素地址,这一点需要记清楚。
首先是如何赋值的问题,这里我写出一个公式:( 类型 (*变量名)[大小] ),所有的指针数组赋值都是按照这个方式来进行的。下代码为示例:
char arr[5];
char (*pa)[5] = &arr;
char* arr[5];
char* (*pa)[5] = &arr;
pa是变量名,*表示pa是指针,[5]表示pa指向的数组是5个元素(char*和char)表示pa指向的数组的元素类型为char*和char。
第二个的取数据顺序为数组指针指向数组地址,数组地址指向数据地址,数据地址指向数据。
下面为对数组指针的使用:(看起来很奇怪,真的!)
int main()
{
int arr[10] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
int (*p)[10] = &arr;
int i = 0;
for (i = 0; i < 10; i++)
{
printf("%d ",*(*p+i));
}
return 0;
}
这里我只对*(*p+i)进行解释,p代表指针指向了arr[10]这个数组,但是如果我们直接对它加一操作,那么会得到位移整个int arr[10]大小的地址值,所以我们需要解引用,进入arr这个地址,拿到整个数组的地址,如何加i位移,最后再*拿到其中地址对应的数据。
对二维数组的使用:
//当作为二维数组传参时,传入的参数的地址为第0行的地址
//所以函数接收时写为类型(*变量名)[第0行的长度]
void print2(int(*pa)[5], int x, int y)
{
int i = 0;
for (i = 0; i < x; i++)
{
int j = 0;
for (j = 0; j < y; j++)
{
//写法一:
printf("%d ",*(*(pa+i)+j));
//pa+i:表示第i行的元素地址
//*(pa+i):表示进入第i行的首地址,存的也是地址
//(*(pa+i)+j):表示第i行里面数据的地址向后位移指向j位
//*(*(pa+i)+j):表示将第i行第j位的数据的值取出来
//然后就是对应的数字
//写法二:
printf("%d ", (*(pa + i))[j]);
//pa+i:表示第i行的元素地址
//*(pa+i):表示进入第i行的首地址,存的也是地址
//(*(pa + i)):可以理解为一维数组的数组名
//[j]:表示为第j位的数据
//等效:(*(pa + i)) = arr,[j] = [j],应用方式是一样的
//arr[i] = *(arr+i)
//写法三:
printf("%d ", pa[i][j]);
}
printf("\n");
}
}
int main()
{
int arr[3][5] = { { 1, 2, 3, 4, 5 }, { 2, 2, 3, 4, 5 }, { 3, 4, 5, 6, 7 } };
print2(arr,3,5);
return 0;
}
对于二维数组,利用数组指针调用时需要记住,作为参数传入函数时,函数拿到的其实是二维数组第一行的地址,而不是第一行第一个的元素地址。那么我们的函数接受的参数就只需要定义为( 类型 (*变量名)[列大小] )。
其中的写法一二三我都是按照一个转换公式写的,那就是pa[i] = *(pa+1),二维那就是pa[i][j] = *(*(pa+i)+j),也是可以拆分的,你们可以试一试。
以上就是全部内容。由于我也是初学,也只学会这些内容,相当于我对这节内容的心得,希望能帮到你们。