C语言指针详解二
- 对数组名的理解
- 数组名就是数组首元素(第一个元素)的地址。
int* p=arr[0];
int* q=arr;
//指针p与指针q是完全相同的
- &arr,这里的数组名表示整个数组,取出的是整个数组的地址(整个数组的地址和数组首元素的地址是有区别的),尽管他们在数值上大小相同。
int* p=arr;
int* q=&arr;
//p+1为跳过该数组的一个元素
//q+1为跳过整个数组
注意:sizeof为特殊情况,sizeof(arr) 为计算整个数组的大小,单位为字节(B)。
- 指针访问数组
#include<stdio.h>
int main()
{
int arr[5] = { 0 };
int sz = sizeof(arr) / sizeof(arr[0]);
int* p = arr;
//指针加减整数的应用
for (int i = 0; i < sz; i++) {
scanf("%d", p + i);
}
//指针解引用的应用
for (int j = 0; j < sz; j++) {
printf("%d ", *(p + j));
}
return 0;
}
注意:在第13行中,将 *printf("%d ", (p + j)); 换成 printf("%d ", p[j]); 也可以正常打印。
同理 arr[i] 应该等价于*(arr+i),数组元素的访问在编译器处理的时候,也是转换成首元素的地址+偏移量求出元素的地址,然后解引用来访问的。
- 一维数组传参的本质
在数组传参的时候,传递的是数组名,也就是说本质上数组传参本质上传递的是数组首元素的地址,所以在函数内部是没办法求的数组元素个数的。
#include<stdio.h>
void test1(int arr[])//写为数组形式
{
int k = sizeof(arr) / sizeof(arr[0]);
printf("%d", k);
}
void test2(int* arr)//写为指针形式
{
int k = sizeof(arr) / sizeof(arr[0]);
printf("%d", k);
}
//test1函数与test2函数等价
int main()
{
int arr[10] = { 1,2,3,4,5,6,7,8,9,0 };
test1(arr);
test2(arr);
return 0;
}
输出结果:在x64环境下为2,在x86环境下为1.
总结:⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式。
- 二级指针
指针变量也是变量,指针变量的地址存放于二级指针中。
- 指针数组
我们可以类比于字符数组是用来存放字符的,那么指针数组就是用来存放指针变量的,指针数组的每个元素是地址,又可以指向一块区域。
- 用指针数组模拟二维数组
#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数组中
int* parr[] = { arr1,arr2,arr3 };
//第一层循环为选择parr中的元素(arr1、arr2或arr3)
for (int i = 0; i < 3; i++) {
//第二层循环为选择arri中的元素(1、2、3......)
for (int j = 0; j < 5; j++) {
printf("%d ", parr[i][j]);
}
printf("\n");
}
return 0;
}
- 注意:parr[ i ]是访问parr数组的元素,parr[ i ]找到的数组元素指向了整型一维数组(arri),parr[ i ][ j ]就是整型⼀维数组中的元素。
但是上述的代码模拟出二维数组的效果,实际上并非完全是二维数组,因为每一行在内存中,不是连续储存的。