目录
1. 数组名的理解
这⾥我们使⽤ &arr[0] 的⽅式拿到了数组第⼀个元素的地址,但是其实数组名本来就是地址,⽽且是数组⾸元素的地址。
#include <stdio.h>
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
printf("&arr[0] = %p\n", &arr[0]);
printf("arr = %p\n", arr);
return 0;
}
一般情况下数组名就是首元素地址,但是以下两种情况特殊:
(1) sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩,单位是字节。
(2)&数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素的地址是有区别的,如果直接用(数组名+1),则跳过的大小是数组中元素的大小。若用&数组名+1,则跳过的大小是整个数组的大小)
2. 使⽤指针访问数组
若使用指针的方式访问数组,例如:
#include <stdio.h>
int main()
{
int arr[10] = {0};
//输⼊
int i = 0;
int sz = sizeof(arr)/sizeof(arr[0]);
//输⼊
int* p = arr;
for(i=0; i<sz; i++)
{
scanf("%d", p+i);
//scanf("%d", arr+i);//也可以这样写
}
//输出
for(i=0; i<sz; i++)
{
printf("%d ", *(p+i));
}
return 0;
}
上述代码也能打印数组中的值,所以可知arr[i] == *(p + i) == p[i] .
3. ⼀维数组传参的本质
#include <stdio.h>
void test(int arr[])
{
int sz2 = sizeof(arr)/sizeof(arr[0]);
printf("sz2 = %d\n", sz2);
}
int main()
{
int arr[10] = {1,2,3,4,5,6,7,8,9,10};
int sz1 = sizeof(arr)/sizeof(arr[0]);
printf("sz1 = %d\n", sz1);
test(arr);
return 0;
}
数组名是首元素的地址;那么在数组传参的时候,传递的是数组名,也就是说本质上数组传参传递的是数组首元素的地址。所以上述函数中的代码计算的是首元素/元素大小 = 一个元素。所以在函数内部是没办法求得数组元素个数的。
总结:一维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式来接收。
4. ⼆级指针
指针变量也是变量,是变量就有地址,存放指针变量地址的指针变量就是二级指针。
二级指针解引用找到的是一级指针。二级指针解引用两次才能找到一级指针所指向的内容。
6. 指针数组
类比一下,整型数组,是存放整型的数组,字符数组是存放字符的数组。那么指针数组,是存放指针的数组。指针数组的每个元素都是用来存放地址(指针)的。
注:指针数组是数组,用于存放指针。
指针数组的形式和整型数组类似:
7. 指针数组模拟⼆维数组
#include <stdio.h>
int main()
{
//创建三个数组
int arr1[] = {1,2,3,4,5};
int arr2[] = {2,3,4,5,6};
int arr3[] = {3,4,5,6,7};
//创建指针数组,用于存放指向上述三个数组的指针
int* parr[3] = {arr1, arr2, arr3};
int i = 0;
int j = 0;
//打印二维数组
for (i = 0; i < 3; i++)
{
for (j = 0; j < 5; j++)
{
printf("%d ", parr[i][j]);
}
printf("\n");
}
return 0;
}
parr[i]是访问parr数组的元素,parr[i]找到的数组元素指向了整型一维数组,parr[i][j]就是整型一维数组i中第j个元素。上述的代码模拟出二维数组的效果,但是并非完全是二维数组,因为C语言中真正的二维数组每一行再内存中存储的地址是连续的,上述的每一行并非是连续的。