一、 数组指针的概念
前言:我们之前只是顾名思义的思考了数组指针的概念,它是强调的指针。接下来深刻的学习数组指针
这个内容是与指针数组容易混淆的
接下来通过类比法学习
int *p=NULL;//p是整型指针-指向整型的指针-可以存放整型的地址
char *pc=NULL;//pc是字符指针-指向字符的指针-可以存放字符的地址
由此我们可以联想数组指针是什么?
我想应该是这样的:数组指针-指向数组的指针-存放数组的地址
我们也可以用类比法学习数组指针的类型
int* p1=&arr;//这个指针的类型为int *
char* p2=&arr;//这个指针的类型为char *
int (*p3)[5]=&arr;//这个指针的类型为int(*)[5]
我想应该是这样的:指针的类型就是指针声明中去掉数组名。
这个与我们那个数组的变量类型有点点相似。
补充:
对于int arr[10];
arr-首元素地址
&arr[0]-首元素的地址
&arr-数组的地址
二、指针数组与数组指针在写法的区分
用两个常见代码进行区分:
int arr1[10]={1,2,3,4,5,6,7,8,9,10};
int *p[10]=&arr;
//上面的这个是指针数组
int arr2[10]={1,2,3,4,5,6,7,8,9,10};
int (*p)[10]=&arr
//上面的这个是数组指针
这是最简单的套路区分,当我们遇到大佬写的复杂代码时,这时我们需要理解一下到底是怎么回事
看我们是数组指针还是指针数组,我们首先要知道这两个名词分别强调的是什么,数组指针本质上是指针,而指针数组本质上是数组;所以我们在代码中区分它们的时候往往是看的我们的优先级,例如在int* p[10]中我们操作符’[]’的优先级大于操作符’*'所以它首先是一个数组,由此它是一个指针数组;反知我们用操作符’0‘里面的内容优先级大于’[],它首先是一个指针,所以它是数组指针。
一句话:我们看操作符的优先级
补充:
在int (*p)[10]=&arr;中,在定义变量的时候,这个操作符’*'不是解引用操作符,而是告诉我们这个p是指针。
三、使用数组指针实现二维数组的打印
#define _CRT_SECURE_NO_WARNINGS 1
#pragma warning(disable:6031)
#include<stdio.h>
void print1(int arr[3][4], int x, int y)
{
int i, j;
for (i = 0; i < x; i++)
{
for (j = 0; j < y; j++)
{
printf("%d ", arr[i][j]);
}
printf("\n");
}
}
void print2(int (*p)[4],int x,int y)
{
int i, j;
for (i = 0; i < x; i++)
{
for (j = 0; j < y; j++)
{
printf("%d ", *(*(p+i)+j));
}
printf("\n");
}
}
int main()
{
int arr[3][4] = { {1,2,3,4}, {2,3,4,5},{3,4,5,6} };
print1(arr, 3, 4);
printf("\n");
print2(arr, 3, 4);
return 0;
}
重点理解语句:
printf("%d ", ((p+i)+j));
理解这条语句,我们首先要知道等价语句
int arr[10]={1,2,3,4,5,6,7,8,9,10};
int*p=&arr;
printf("%d",p[i]);
printf("%d",*(p+i));
printf("%d",*(arr+i));
printf("%d",arr[i]);
这几个printf函数里面的内容完全是等价的
arr[i]== p[i] == *(p+i)== *(arr+i)==*(arr+i)==arr[i]
在数组指针中也存在类似的等价
补充:
二维数组传参我们也是传的是首元素的地址,只不过我们需要将二维数组想象成一维数组,所以我们的每一行就是一个一维数组,我们传参的时候,传过去的首元素地址是第一行元素的地址
四、结语
学完数组指针和指针数组,我们用一个题目总结一下;
第一个:创建的是数组,数组里面有5个元素,每个元素的类型是int
第二个:创建的是数组,数组里面有10个元素,每个元素的类型是int *,paar1是指针数组
第三个:创建的是指针,指针指向有10个元素的数组,每个元素的类型是int,parr2是数组指针
第四个:创建的是数组,数组里面有10个元素,每个元素都是一个数组指针,数组指针指向有5个元素的数组,每个元素的类型是int。