1.数组指针和指针数组
1.1数组指针
先来看一段代码,p1和p2分别是什么?
int* p1[5]
int (*p2)[5]
这里要注意运算符的优先级,[ ]的优先级比*的优先级要高
第一行代码:p1先与[5]结合之后*再与p1[5]结合,形成指针数组,类型是int *[5],数组名是p1,包含有指向int类型的5个指针变量。
第二行代码:p2先和*结合,说明p2是⼀个指针变量变量,然后指着指向的是⼀个⼤⼩为10个整型的数组。所以p2是⼀个指针,指向⼀个数组,叫数组指针。但是并没有对p2初始化且p2只能指向含有五个int型元素的数组。
1.2数组指针的初始化
如果要存放数组的地址,那么就要用到数组指针
int (*p) [10] = &arr;
| | |
| | |
| | p指向数组的元素个数
| p是数组指针变量名
p指向的数组的元素类型
运行以上代码后可以看到&arr和p的地址和类型是一样的。
2.二维数组传参
⾸先我们再次理解⼀下⼆维数组,⼆维数组起始可以看做是每个元素是⼀维数组的数组,也就是⼆维数组的每个元素是⼀个⼀维数组。那么⼆维数组的⾸元素就是第⼀⾏,是个⼀维数组。

数组名本身代表的是数组首元素的地址,但是有两个例外:
sizeof(数组名)&数组名这两者中arr代表的都是整个数组,其他情况下arr代表的都是首元素的地址
函数传参也不例外,
void test(int arr[3][5])
{
//函数传参中传入的也是二维数组首元素的地址
}
int main()
{
int arr[3][5]={0};
test(arr);//二维数组的首地址其实是首行一维数组的地址
}
所以,根据数组名是数组⾸元素的地址这个规则,⼆维数组的数组名表⽰的就是第⼀⾏的地址,是⼀ 维数组的地址。根据上⾯的例⼦,第⼀⾏的⼀维数组的类型就是 int [5] ,所以第⼀⾏的地址类
型就是数组指针类型 int(*)[5] 。那就意味着⼆维数组传参本质上也是传递了地址,传递的是第⼀
⾏这个⼀维数组的地址,那么形参也是可以写成指针形式的。如下:
void test(int(*p)[5])
{
}
int main()
{
int arr[3][5]={0};
test(arr);
}
总结:⼆维数组传参,形参的部分可以写成数组,也可以写成指针形式。